mirror of
https://github.com/squidfunk/mkdocs-material.git
synced 2024-06-14 11:52:32 +03:00
59e851a4c1
Closes #1717
1 line
301 KiB
Plaintext
1 line
301 KiB
Plaintext
{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/assets/javascripts/browser/document/index.ts","webpack:///./src/assets/javascripts/browser/element/_/index.ts","webpack:///./src/assets/javascripts/browser/element/focus/index.ts","webpack:///./src/assets/javascripts/browser/element/offset/index.ts","webpack:///./src/assets/javascripts/browser/element/select/index.ts","webpack:///./src/assets/javascripts/browser/element/size/index.ts","webpack:///./src/assets/javascripts/browser/keyboard/index.ts","webpack:///./src/assets/javascripts/browser/location/_/index.ts","webpack:///./src/assets/javascripts/browser/location/base/index.ts","webpack:///./src/assets/javascripts/browser/location/hash/index.ts","webpack:///./src/assets/javascripts/browser/media/index.ts","webpack:///./src/assets/javascripts/browser/toggle/index.ts","webpack:///./src/assets/javascripts/browser/viewport/offset/index.ts","webpack:///./src/assets/javascripts/browser/viewport/size/index.ts","webpack:///./src/assets/javascripts/browser/viewport/_/index.ts","webpack:///./src/assets/javascripts/browser/worker/index.ts","webpack:///./src/assets/javascripts/utilities/config/index.ts","webpack:///./src/assets/javascripts/utilities/jsx/index.ts","webpack:///./src/assets/javascripts/utilities/rxjs/index.ts","webpack:///./src/assets/javascripts/utilities/string/index.ts","webpack:///./src/assets/javascripts/components/index.ts","webpack:///./src/assets/javascripts/integrations/clipboard/index.ts","webpack:///./src/assets/javascripts/integrations/dialog/index.ts","webpack:///./src/assets/javascripts/integrations/instant/index.ts","webpack:///./src/assets/javascripts/integrations/keyboard/index.ts","webpack:///./src/assets/javascripts/components/_/index.ts","webpack:///./src/assets/javascripts/components/toc/anchor/set/index.ts","webpack:///./src/assets/javascripts/components/shared/index.ts","webpack:///./src/assets/javascripts/templates/clipboard/index.tsx","webpack:///./src/assets/javascripts/templates/search/index.tsx","webpack:///./src/assets/javascripts/templates/source/index.tsx","webpack:///./src/assets/javascripts/templates/table/index.tsx","webpack:///./src/assets/javascripts/components/shared/sidebar/set/index.ts","webpack:///./src/assets/javascripts/components/toc/anchor/index.ts","webpack:///./src/assets/javascripts/integrations/search/_/index.ts","webpack:///./src/assets/javascripts/integrations/search/document/index.ts","webpack:///./src/assets/javascripts/integrations/search/highlighter/index.ts","webpack:///./src/assets/javascripts/integrations/search/transform/index.ts","webpack:///./src/assets/javascripts/integrations/search/worker/message/index.ts","webpack:///./src/assets/javascripts/integrations/search/worker/_/index.ts","webpack:///./src/assets/javascripts/components/shared/sidebar/index.ts","webpack:///./src/assets/javascripts/components/shared/sidebar/react/index.ts","webpack:///./src/assets/javascripts/components/toc/index.ts","webpack:///./src/assets/javascripts/components/toc/_/index.ts","webpack:///./src/assets/javascripts/components/toc/anchor/react/index.ts","webpack:///./src/assets/javascripts/components/search/_/index.ts","webpack:///./src/assets/javascripts/components/search/query/_/index.ts","webpack:///./src/assets/javascripts/components/search/query/react/index.ts","webpack:///./src/assets/javascripts/components/search/reset/_/index.ts","webpack:///./src/assets/javascripts/components/search/reset/react/index.ts","webpack:///./src/assets/javascripts/components/search/result/set/index.ts","webpack:///./src/assets/javascripts/components/search/result/react/index.ts","webpack:///./src/assets/javascripts/components/search/result/_/index.ts","webpack:///./src/assets/javascripts/components/main/_/index.ts","webpack:///./src/assets/javascripts/components/main/react/index.ts","webpack:///./src/assets/javascripts/components/main/set/index.ts","webpack:///./src/assets/javascripts/components/hero/_/index.ts","webpack:///./src/assets/javascripts/components/hero/react/index.ts","webpack:///./src/assets/javascripts/components/hero/set/index.ts","webpack:///./src/assets/javascripts/components/header/_/index.ts","webpack:///./src/assets/javascripts/components/header/react/index.ts","webpack:///./src/assets/javascripts/components/header/set/index.ts","webpack:///./src/assets/javascripts/components/tabs/_/index.ts","webpack:///./src/assets/javascripts/components/tabs/react/index.ts","webpack:///./src/assets/javascripts/components/tabs/set/index.ts","webpack:///./src/assets/javascripts/components/navigation/_/index.ts","webpack:///./src/assets/javascripts/patches/scrollfix/index.ts","webpack:///./src/assets/javascripts/patches/source/index.ts","webpack:///./src/assets/javascripts/patches/source/github/index.ts","webpack:///./src/assets/javascripts/patches/source/gitlab/index.ts","webpack:///./src/assets/javascripts/index.ts","webpack:///./src/assets/javascripts/patches/code/index.ts","webpack:///./src/assets/javascripts/patches/details/index.ts","webpack:///./src/assets/javascripts/patches/script/index.ts","webpack:///./src/assets/javascripts/patches/table/index.ts"],"names":["webpackJsonpCallback","data","moduleId","chunkId","chunkIds","moreModules","executeModules","i","resolves","length","Object","prototype","hasOwnProperty","call","installedChunks","push","modules","parentJsonpFunction","shift","deferredModules","apply","checkDeferredModules","result","deferredModule","fulfilled","j","depId","splice","__webpack_require__","s","installedModules","0","exports","module","l","m","c","d","name","getter","o","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","object","property","p","jsonpArray","window","oldJsonpFunction","slice","watchDocument","document$","ReplaySubject","fromEvent","document","pipe","mapTo","subscribe","getElement","selector","node","querySelector","undefined","getElementOrThrow","el","ReferenceError","getActiveElement","activeElement","HTMLElement","getElements","Array","from","querySelectorAll","createElement","tagName","replaceElement","source","target","replaceWith","setElementFocus","focus","blur","watchElementFocus","merge","map","startWith","shareReplay","getElementOffset","x","scrollLeft","y","scrollTop","watchElementOffset","setElementSelection","HTMLInputElement","Error","select","watchElementSize","fromEventPattern","next","contentRect","width","Math","round","height","observe","offsetWidth","offsetHeight","getElementSize","isSusceptibleToKeyboard","isContentEditable","watchKeyboard","filter","ev","metaKey","ctrlKey","type","claim","preventDefault","stopPropagation","share","setLocation","url","location","href","isLocalLocation","ref","host","test","pathname","isAnchorLocation","hash","watchLocation","BehaviorSubject","URL","watchLocationBase","base","take","toString","replace","getLocationHash","substring","setLocationHash","addEventListener","click","watchLocationHash","watchMedia","query","media","matchMedia","addListener","matches","toggles","drawer","search","getToggle","checked","setToggle","watchToggle","getViewportOffset","max","pageXOffset","pageYOffset","setViewportOffset","scrollTo","getViewportSize","innerWidth","innerHeight","watchViewport","combineLatest","passive","offset","size","watchViewportAt","header$","viewport$","size$","distinctUntilKeyChanged","offset$","offsetLeft","offsetTop","watchWorker","worker","tx$","rx$","pluck","throttle","leading","trailing","tap","message","postMessage","switchMapTo","isConfig","config","features","createElementNS","setAttribute","setAttributeNS","appendChild","child","innerHTML","Node","isArray","h","attributes","keys","attr","children","cache","factory","defer","sessionStorage","getItem","of","JSON","parse","value$","setItem","stringify","err","lang","translate","textContent","truncate","toFixed","len","charCodeAt","setupClipboard","dialog$","forEach","block","index","parent","parentElement","id","insertBefore","clipboard$","on","clearSelection","setupDialog","duration","Subject","dialog","classList","add","switchMap","text","body","container","observeOn","animationFrame","delay","removeAttribute","remove","setupInstantLoading","urls","location$","history","scrollRestoration","favicon","state$","closest","includes","push$","pop$","state","distinctUntilChanged","prev","ajax$","skip","ajax","responseType","withCredentials","catchError","sample","pushState","dom","DOMParser","response","parseFromString","instant$","withLatestFrom","title","head","dispatchEvent","CustomEvent","debounceTime","replaceState","bufferCount","setupKeyboard","keyboard$","active","els","indexOf","components$","setupComponents","names","reduce","components","useComponent","setAnchorBlur","resetAnchorBlur","setAnchorActive","toggle","resetAnchorActive","css","renderClipboardButton","class","xmlns","viewBox","renderSearchResult","article","sections","icon","tabIndex","renderSource","facts","fact","renderTable","table","setSidebarOffset","style","top","resetSidebarOffset","setSidebarHeight","resetSidebarHeight","docs","pipeline","this","documents","Map","doc","split","path","linked","set","setupSearchDocumentMap","highlight","separator","RegExp","_","term","trim","match","setupSearchHighlighter","lunr","reset","fn","use","multiLanguage","field","boost","Index","load","groups","results","section","console","warn","defaultTransform","SearchMessageType","isSearchReadyMessage","READY","isSearchQueryMessage","QUERY","isSearchResultMessage","RESULT","setupSearchIndex","identity","setupSearchWorker","index$","base$","Worker","SETUP","watchSidebar","main$","adjust","min","lock","a","b","applySidebar","mountTableOfContents","tablet$","tablet","sidebar$","anchors$","sidebar","anchors","watchAnchorList","decodeURIComponent","adjust$","header","anchor","pop","applyAnchorList","mountSearch","query$","reset$","result$","status$","status","mountSearchQuery","options","focus$","watchSearchQuery","mountSearchReset","watchSearchReset","addToSearchResultList","applySearchResult","ready$","fetch$","list","meta","setSearchResultMeta","resetSearchResultMeta","scan","scrollHeight","finalize","resetSearchResultList","mountSearchResult","mountMain","setHeaderShadow","resetHeaderShadow","border$","bottom","watchMain","main","mountHero","hidden","setHeroHidden","resetHeroHidden","applyHero","mountHeader","styles","getComputedStyle","position","sticky","watchHeader","type$","hx","setHeaderTitleActive","resetHeaderTitleActive","applyHeaderType","mountTabs","screen$","screen","setTabsHidden","resetTabsHidden","applyTabs","mountNavigation","isAppleDevice","navigator","userAgent","fetchSourceFacts","toLowerCase","user","repo","stargazers_count","forks_count","public_repos","fetchSourceFactsFromGitHub","project","encodeURIComponent","star_count","fetchSourceFactsFromGitLab","setScrollLock","resetScrollLock","parseInt","initialize","SyntaxError","hash$","els$","scrollWidth","clientWidth","patchCodeBlocks","details","open","scrollIntoView","patchDetails","src","script","hasAttribute","patchSource","sentinel","patchTables","iif","patchScrollfix","navigation$","toc$","tabs$","hero$","search$","transform","protocol","sortBy","prop","charAt","visibility","values","documentElement"],"mappings":"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","file":"assets/javascripts/bundle.5f27aba8.min.js","sourcesContent":[" \t// install a JSONP callback for chunk loading\n \tfunction webpackJsonpCallback(data) {\n \t\tvar chunkIds = data[0];\n \t\tvar moreModules = data[1];\n \t\tvar executeModules = data[2];\n\n \t\t// add \"moreModules\" to the modules object,\n \t\t// then flag all \"chunkIds\" as loaded and fire callback\n \t\tvar moduleId, chunkId, i = 0, resolves = [];\n \t\tfor(;i < chunkIds.length; i++) {\n \t\t\tchunkId = chunkIds[i];\n \t\t\tif(Object.prototype.hasOwnProperty.call(installedChunks, chunkId) && installedChunks[chunkId]) {\n \t\t\t\tresolves.push(installedChunks[chunkId][0]);\n \t\t\t}\n \t\t\tinstalledChunks[chunkId] = 0;\n \t\t}\n \t\tfor(moduleId in moreModules) {\n \t\t\tif(Object.prototype.hasOwnProperty.call(moreModules, moduleId)) {\n \t\t\t\tmodules[moduleId] = moreModules[moduleId];\n \t\t\t}\n \t\t}\n \t\tif(parentJsonpFunction) parentJsonpFunction(data);\n\n \t\twhile(resolves.length) {\n \t\t\tresolves.shift()();\n \t\t}\n\n \t\t// add entry modules from loaded chunk to deferred list\n \t\tdeferredModules.push.apply(deferredModules, executeModules || []);\n\n \t\t// run deferred modules when all chunks ready\n \t\treturn checkDeferredModules();\n \t};\n \tfunction checkDeferredModules() {\n \t\tvar result;\n \t\tfor(var i = 0; i < deferredModules.length; i++) {\n \t\t\tvar deferredModule = deferredModules[i];\n \t\t\tvar fulfilled = true;\n \t\t\tfor(var j = 1; j < deferredModule.length; j++) {\n \t\t\t\tvar depId = deferredModule[j];\n \t\t\t\tif(installedChunks[depId] !== 0) fulfilled = false;\n \t\t\t}\n \t\t\tif(fulfilled) {\n \t\t\t\tdeferredModules.splice(i--, 1);\n \t\t\t\tresult = __webpack_require__(__webpack_require__.s = deferredModule[0]);\n \t\t\t}\n \t\t}\n\n \t\treturn result;\n \t}\n\n \t// The module cache\n \tvar installedModules = {};\n\n \t// object to store loaded and loading chunks\n \t// undefined = chunk not loaded, null = chunk preloaded/prefetched\n \t// Promise = chunk loading, 0 = chunk loaded\n \tvar installedChunks = {\n \t\t0: 0\n \t};\n\n \tvar deferredModules = [];\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \tvar jsonpArray = window[\"webpackJsonp\"] = window[\"webpackJsonp\"] || [];\n \tvar oldJsonpFunction = jsonpArray.push.bind(jsonpArray);\n \tjsonpArray.push = webpackJsonpCallback;\n \tjsonpArray = jsonpArray.slice();\n \tfor(var i = 0; i < jsonpArray.length; i++) webpackJsonpCallback(jsonpArray[i]);\n \tvar parentJsonpFunction = oldJsonpFunction;\n\n\n \t// add entry module to deferred list\n \tdeferredModules.push([88,1]);\n \t// run deferred modules when ready\n \treturn checkDeferredModules();\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { ReplaySubject, Subject, fromEvent } from \"rxjs\"\r\nimport { mapTo } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch document\r\n *\r\n * Documents must be implemented as subjects, so all downstream observables are\r\n * automatically updated when a new document is emitted. This enabled features\r\n * like instant loading.\r\n *\r\n * @return Document subject\r\n */\r\nexport function watchDocument(): Subject<Document> {\r\n const document$ = new ReplaySubject<Document>()\r\n fromEvent(document, \"DOMContentLoaded\")\r\n .pipe(\r\n mapTo(document)\r\n )\r\n .subscribe(document$)\r\n\r\n /* Return document */\r\n return document$\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve an element matching the query selector\r\n *\r\n * @template T - Element type\r\n *\r\n * @param selector - Query selector\r\n * @param node - Node of reference\r\n *\r\n * @return Element or nothing\r\n */\r\nexport function getElement<T extends HTMLElement>(\r\n selector: string, node: ParentNode = document\r\n): T | undefined {\r\n return node.querySelector<T>(selector) || undefined\r\n}\r\n\r\n/**\r\n * Retrieve an element matching a query selector or throw a reference error\r\n *\r\n * @template T - Element type\r\n *\r\n * @param selector - Query selector\r\n * @param node - Node of reference\r\n *\r\n * @return Element\r\n */\r\nexport function getElementOrThrow<T extends HTMLElement>(\r\n selector: string, node: ParentNode = document\r\n): T {\r\n const el = getElement<T>(selector, node)\r\n if (typeof el === \"undefined\")\r\n throw new ReferenceError(\r\n `Missing element: expected \"${selector}\" to be present`\r\n )\r\n return el\r\n}\r\n\r\n/**\r\n * Retrieve the currently active element\r\n *\r\n * @return Element or nothing\r\n */\r\nexport function getActiveElement(): HTMLElement | undefined {\r\n return document.activeElement instanceof HTMLElement\r\n ? document.activeElement\r\n : undefined\r\n}\r\n\r\n/**\r\n * Retrieve all elements matching the query selector\r\n *\r\n * @template T - Element type\r\n *\r\n * @param selector - Query selector\r\n * @param node - Node of reference\r\n *\r\n * @return Elements\r\n */\r\nexport function getElements<T extends HTMLElement>(\r\n selector: string, node: ParentNode = document\r\n): T[] {\r\n return Array.from(node.querySelectorAll<T>(selector))\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Create an element\r\n *\r\n * @template T - Tag name type\r\n *\r\n * @param tagName - Tag name\r\n *\r\n * @return Element\r\n */\r\nexport function createElement<\r\n T extends keyof HTMLElementTagNameMap\r\n>(tagName: T): HTMLElementTagNameMap[T] {\r\n return document.createElement(tagName)\r\n}\r\n\r\n/**\r\n * Replace an element with another element\r\n *\r\n * @param source - Source element\r\n * @param target - Target element\r\n */\r\nexport function replaceElement(\r\n source: HTMLElement, target: Node\r\n): void {\r\n source.replaceWith(target)\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent, merge } from \"rxjs\"\r\nimport { map, shareReplay, startWith } from \"rxjs/operators\"\r\n\r\nimport { getActiveElement } from \"../_\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set element focus\r\n *\r\n * @param el - Element\r\n * @param value - Whether the element should be focused\r\n */\r\nexport function setElementFocus(\r\nel: HTMLElement, value: boolean = true\r\n): void {\r\n if (value)\r\n el.focus()\r\n else\r\n el.blur()\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch element focus\r\n *\r\n * @param el - Element\r\n *\r\n * @return Element focus observable\r\n */\r\nexport function watchElementFocus(\r\n el: HTMLElement\r\n): Observable<boolean> {\r\n return merge(\r\n fromEvent<FocusEvent>(el, \"focus\"),\r\n fromEvent<FocusEvent>(el, \"blur\")\r\n )\r\n .pipe(\r\n map(({ type }) => type === \"focus\"),\r\n startWith(el === getActiveElement()),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent, merge } from \"rxjs\"\r\nimport { map, shareReplay, startWith } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Element offset\r\n */\r\nexport interface ElementOffset {\r\n x: number /* Horizontal offset */\r\n y: number /* Vertical offset */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve element offset\r\n *\r\n * @param el - Element\r\n *\r\n * @return Element offset\r\n */\r\nexport function getElementOffset(el: HTMLElement): ElementOffset {\r\n return {\r\n x: el.scrollLeft,\r\n y: el.scrollTop\r\n }\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch element offset\r\n *\r\n * @param el - Element\r\n *\r\n * @return Element offset observable\r\n */\r\nexport function watchElementOffset(\r\n el: HTMLElement\r\n): Observable<ElementOffset> {\r\n return merge(\r\n fromEvent(el, \"scroll\"),\r\n fromEvent(window, \"resize\")\r\n )\r\n .pipe(\r\n map(() => getElementOffset(el)),\r\n startWith(getElementOffset(el)),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set element text selection\r\n *\r\n * @param el - Element\r\n */\r\nexport function setElementSelection(\r\n el: HTMLElement\r\n): void {\r\n if (el instanceof HTMLInputElement)\r\n el.select()\r\n else\r\n throw new Error(\"Not implemented\")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport ResizeObserver from \"resize-observer-polyfill\"\r\nimport { Observable, fromEventPattern } from \"rxjs\"\r\nimport { shareReplay, startWith } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Element offset\r\n */\r\nexport interface ElementSize {\r\n width: number /* Element width */\r\n height: number /* Element height */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve element size\r\n *\r\n * @param el - Element\r\n *\r\n * @return Element size\r\n */\r\nexport function getElementSize(el: HTMLElement): ElementSize {\r\n return {\r\n width: el.offsetWidth,\r\n height: el.offsetHeight\r\n }\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch element size\r\n *\r\n * @param el - Element\r\n *\r\n * @return Element size observable\r\n */\r\nexport function watchElementSize(\r\n el: HTMLElement\r\n): Observable<ElementSize> {\r\n return fromEventPattern<ElementSize>(next => {\r\n new ResizeObserver(([{ contentRect }]) => next({\r\n width: Math.round(contentRect.width),\r\n height: Math.round(contentRect.height)\r\n }))\r\n .observe(el)\r\n })\r\n .pipe(\r\n startWith(getElementSize(el)),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent } from \"rxjs\"\r\nimport { filter, map, share } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Key\r\n */\r\nexport interface Key {\r\n type: string /* Key type */\r\n claim(): void /* Key claim */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Check whether an element may receive keyboard input\r\n *\r\n * @param el - Element\r\n *\r\n * @return Test result\r\n */\r\nexport function isSusceptibleToKeyboard(el: HTMLElement): boolean {\r\n switch (el.tagName) {\r\n\r\n /* Form elements */\r\n case \"INPUT\":\r\n case \"SELECT\":\r\n case \"TEXTAREA\":\r\n return true\r\n\r\n /* Everything else */\r\n default:\r\n return el.isContentEditable\r\n }\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch keyboard\r\n *\r\n * @return Keyboard observable\r\n */\r\nexport function watchKeyboard(): Observable<Key> {\r\n return fromEvent<KeyboardEvent>(window, \"keydown\")\r\n .pipe(\r\n filter(ev => !(ev.metaKey || ev.ctrlKey)),\r\n map(ev => ({\r\n type: ev.key,\r\n claim() {\r\n ev.preventDefault()\r\n ev.stopPropagation()\r\n }\r\n })),\r\n share()\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { BehaviorSubject, Subject } from \"rxjs\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve location\r\n *\r\n * This function will return a `URL` object (and not `Location`) in order to\r\n * normalize typings across the application. Furthermore, locations need to be\r\n * tracked without setting them and `Location` is a singleton which represents\r\n * the current location.\r\n *\r\n * @return URL\r\n */\r\nexport function getLocation(): URL {\r\n return new URL(location.href)\r\n}\r\n\r\n/**\r\n * Set location\r\n *\r\n * @param url - URL to change to\r\n */\r\nexport function setLocation(url: URL): void {\r\n location.href = url.href\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Check whether a URL is a local link or a file (except `.html`)\r\n *\r\n * @param url - URL or HTML anchor element\r\n * @param ref - Reference URL\r\n *\r\n * @return Test result\r\n */\r\nexport function isLocalLocation(\r\n url: URL | HTMLAnchorElement,\r\n ref: URL | Location = location\r\n): boolean {\r\n return url.host === ref.host\r\n && /^(?:\\/[\\w-]+)*(?:\\/?|\\.html)$/i.test(url.pathname)\r\n}\r\n\r\n/**\r\n * Check whether a URL is an anchor link on the current page\r\n *\r\n * @param url - URL or HTML anchor element\r\n * @param ref - Reference URL\r\n *\r\n * @return Test result\r\n */\r\nexport function isAnchorLocation(\r\n url: URL | HTMLAnchorElement,\r\n ref: URL | Location = location\r\n): boolean {\r\n return url.pathname === ref.pathname\r\n && url.hash.length > 0\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch location\r\n *\r\n * @return Location subject\r\n */\r\nexport function watchLocation(): Subject<URL> {\r\n return new BehaviorSubject<URL>(getLocation())\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable } from \"rxjs\"\r\nimport { map, shareReplay, take } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch options\r\n */\r\ninterface WatchOptions {\r\n location$: Observable<URL> /* Location observable */\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch location base\r\n *\r\n * @return Location base observable\r\n */\r\nexport function watchLocationBase(\r\n base: string, { location$ }: WatchOptions\r\n): Observable<string> {\r\n return location$\r\n .pipe(\r\n take(1),\r\n map(({ href }) => new URL(base, href)\r\n .toString()\r\n .replace(/\\/$/, \"\")\r\n ),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent } from \"rxjs\"\r\nimport { filter, map, share, startWith } from \"rxjs/operators\"\r\n\r\nimport { createElement } from \"browser\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve location hash\r\n *\r\n * @return Location hash\r\n */\r\nexport function getLocationHash(): string {\r\n return location.hash.substring(1)\r\n}\r\n\r\n/**\r\n * Set location hash\r\n *\r\n * Setting a new fragment identifier via `location.hash` will have no effect\r\n * if the value doesn't change. When a new fragment identifier is set, we want\r\n * the browser to target the respective element at all times, which is why we\r\n * use this dirty little trick.\r\n *\r\n * @param hash - Location hash\r\n */\r\nexport function setLocationHash(hash: string): void {\r\n const el = createElement(\"a\")\r\n el.href = hash\r\n el.addEventListener(\"click\", ev => ev.stopPropagation())\r\n el.click()\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch location hash\r\n *\r\n * @return Location hash observable\r\n */\r\nexport function watchLocationHash(): Observable<string> {\r\n return fromEvent<HashChangeEvent>(window, \"hashchange\")\r\n .pipe(\r\n map(getLocationHash),\r\n startWith(getLocationHash()),\r\n filter(hash => hash.length > 0),\r\n share()\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEventPattern } from \"rxjs\"\r\nimport { shareReplay, startWith } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch media query\r\n *\r\n * @param query - Media query\r\n *\r\n * @return Media observable\r\n */\r\nexport function watchMedia(query: string): Observable<boolean> {\r\n const media = matchMedia(query)\r\n return fromEventPattern<boolean>(next =>\r\n media.addListener(() => next(media.matches))\r\n )\r\n .pipe(\r\n startWith(media.matches),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent } from \"rxjs\"\r\nimport { map, startWith } from \"rxjs/operators\"\r\n\r\nimport { getElementOrThrow } from \"../element\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Toggle\r\n */\r\nexport type Toggle =\r\n | \"drawer\" /* Toggle for drawer */\r\n | \"search\" /* Toggle for search */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Toggle map\r\n */\r\nconst toggles: Record<Toggle, HTMLInputElement> = {\r\n drawer: getElementOrThrow(`[data-md-toggle=drawer]`),\r\n search: getElementOrThrow(`[data-md-toggle=search]`)\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve the value of a toggle\r\n *\r\n * @param name - Toggle\r\n *\r\n * @return Toggle value\r\n */\r\nexport function getToggle(name: Toggle): boolean {\r\n return toggles[name].checked\r\n}\r\n\r\n/**\r\n * Set toggle\r\n *\r\n * Simulating a click event seems to be the most cross-browser compatible way\r\n * of changing the value while also emitting a `change` event. Before, Material\r\n * used `CustomEvent` to programmatically change the value of a toggle, but this\r\n * is a much simpler and cleaner solution which doesn't require a polyfill.\r\n *\r\n * @param name - Toggle\r\n * @param value - Toggle value\r\n */\r\nexport function setToggle(name: Toggle, value: boolean): void {\r\n if (toggles[name].checked !== value)\r\n toggles[name].click()\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch toggle\r\n *\r\n * @param name - Toggle\r\n *\r\n * @return Toggle value observable\r\n */\r\nexport function watchToggle(name: Toggle): Observable<boolean> {\r\n const el = toggles[name]\r\n return fromEvent(el, \"change\")\r\n .pipe(\r\n map(() => el.checked),\r\n startWith(el.checked)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent, merge } from \"rxjs\"\r\nimport { map, startWith } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Viewport offset\r\n */\r\nexport interface ViewportOffset {\r\n x: number /* Horizontal offset */\r\n y: number /* Vertical offset */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve viewport offset\r\n *\r\n * On iOS Safari, viewport offset can be negative due to overflow scrolling.\r\n * As this may induce strange behaviors downstream, we'll just limit it to 0.\r\n *\r\n * @return Viewport offset\r\n */\r\nexport function getViewportOffset(): ViewportOffset {\r\n return {\r\n x: Math.max(0, pageXOffset),\r\n y: Math.max(0, pageYOffset)\r\n }\r\n}\r\n\r\n/**\r\n * Set viewport offset\r\n *\r\n * @param offset - Viewport offset\r\n */\r\nexport function setViewportOffset(\r\n { x, y }: Partial<ViewportOffset>\r\n): void {\r\n window.scrollTo(x || 0, y || 0)\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch viewport offset\r\n *\r\n * @return Viewport offset observable\r\n */\r\nexport function watchViewportOffset(): Observable<ViewportOffset> {\r\n return merge(\r\n fromEvent(window, \"scroll\", { passive: true }),\r\n fromEvent(window, \"resize\", { passive: true })\r\n )\r\n .pipe(\r\n map(getViewportOffset),\r\n startWith(getViewportOffset())\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent } from \"rxjs\"\r\nimport { map, startWith } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Viewport size\r\n */\r\nexport interface ViewportSize {\r\n width: number /* Viewport width */\r\n height: number /* Viewport height */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Retrieve viewport size\r\n *\r\n * @return Viewport size\r\n */\r\nexport function getViewportSize(): ViewportSize {\r\n return {\r\n width: innerWidth,\r\n height: innerHeight\r\n }\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch viewport size\r\n *\r\n * @return Viewport size observable\r\n */\r\nexport function watchViewportSize(): Observable<ViewportSize> {\r\n return fromEvent(window, \"resize\", { passive: true })\r\n .pipe(\r\n map(getViewportSize),\r\n startWith(getViewportSize())\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, combineLatest } from \"rxjs\"\r\nimport {\r\n distinctUntilKeyChanged,\r\n map,\r\n shareReplay\r\n} from \"rxjs/operators\"\r\n\r\nimport { Header } from \"components\"\r\n\r\nimport {\r\n ViewportOffset,\r\n watchViewportOffset\r\n} from \"../offset\"\r\nimport {\r\n ViewportSize,\r\n watchViewportSize\r\n} from \"../size\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Viewport\r\n */\r\nexport interface Viewport {\r\n offset: ViewportOffset /* Viewport offset */\r\n size: ViewportSize /* Viewport size */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch at options\r\n */\r\ninterface WatchAtOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch viewport\r\n *\r\n * @return Viewport observable\r\n */\r\nexport function watchViewport(): Observable<Viewport> {\r\n return combineLatest([\r\n watchViewportOffset(),\r\n watchViewportSize()\r\n ])\r\n .pipe(\r\n map(([offset, size]) => ({ offset, size })),\r\n shareReplay(1)\r\n )\r\n}\r\n\r\n/**\r\n * Watch viewport relative to element\r\n *\r\n * @param el - Element\r\n * @param options - Options\r\n *\r\n * @return Viewport observable\r\n */\r\nexport function watchViewportAt(\r\n el: HTMLElement, { header$, viewport$ }: WatchAtOptions\r\n): Observable<Viewport> {\r\n const size$ = viewport$\r\n .pipe(\r\n distinctUntilKeyChanged(\"size\")\r\n )\r\n\r\n /* Compute element offset */\r\n const offset$ = combineLatest([size$, header$])\r\n .pipe(\r\n map((): ViewportOffset => ({\r\n x: el.offsetLeft,\r\n y: el.offsetTop\r\n }))\r\n )\r\n\r\n /* Compute relative viewport, return hot observable */\r\n return combineLatest([header$, viewport$, offset$])\r\n .pipe(\r\n map(([{ height }, { offset, size }, { x, y }]) => ({\r\n offset: {\r\n x: offset.x - x,\r\n y: offset.y - y + height\r\n },\r\n size\r\n })),\r\n shareReplay(1)\r\n )\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { Observable, Subject, fromEventPattern } from \"rxjs\"\nimport {\n pluck,\n share,\n switchMapTo,\n tap,\n throttle\n} from \"rxjs/operators\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Worker message\n */\nexport interface WorkerMessage {\n type: unknown /* Message type */\n data?: unknown /* Message data */\n}\n\n/**\n * Worker handler\n *\n * @template T - Message type\n */\nexport interface WorkerHandler<\n T extends WorkerMessage\n> {\n tx$: Subject<T> /* Message transmission subject */\n rx$: Observable<T> /* Message receive observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch options\n *\n * @template T - Worker message type\n */\ninterface WatchOptions<T extends WorkerMessage> {\n tx$: Observable<T> /* Message transmission observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch a web worker\n *\n * This function returns an observable that will send all values emitted by the\n * message observable to the web worker. Web worker communication is expected\n * to be bidirectional (request-response) and synchronous. Messages that are\n * emitted during a pending request are throttled, the last one is emitted.\n *\n * @param worker - Web worker\n * @param options - Options\n *\n * @return Worker message observable\n */\nexport function watchWorker<T extends WorkerMessage>(\n worker: Worker, { tx$ }: WatchOptions<T>\n): Observable<T> {\n\n /* Intercept messages from worker-like objects */\n const rx$ = fromEventPattern<MessageEvent>(next =>\n worker.addEventListener(\"message\", next)\n )\n .pipe<T>(\n pluck(\"data\")\n )\n\n /* Send and receive messages, return hot observable */\n return tx$\n .pipe(\n throttle(() => rx$, { leading: true, trailing: true }),\n tap(message => worker.postMessage(message)),\n switchMapTo(rx$),\n share()\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { SearchIndex, SearchTransformFn } from \"integrations\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Feature flags\r\n */\r\nexport type Feature =\r\n | \"tabs\" /* Tabs navigation */\r\n | \"instant\" /* Instant loading\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Configuration\r\n */\r\nexport interface Config {\r\n base: string /* Base URL */\r\n features: Feature[] /* Feature flags */\r\n search: {\r\n worker: string /* Worker URL */\r\n index?: Promise<SearchIndex> /* Promise resolving with index */\r\n transform?: SearchTransformFn /* Transformation function */\r\n }\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Ensure that the given value is a valid configuration\r\n *\r\n * We could use `jsonschema` or any other schema validation framework, but that\r\n * would just add more bloat to the bundle, so we'll keep it plain and simple.\r\n *\r\n * @param config - Configuration\r\n *\r\n * @return Test result\r\n */\r\nexport function isConfig(config: any): config is Config {\r\n return typeof config === \"object\"\r\n && typeof config.base === \"string\"\r\n && typeof config.features === \"object\"\r\n && typeof config.search === \"object\"\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n// tslint:disable no-null-keyword\r\n\r\nimport { JSX as JSXInternal } from \"preact\"\r\nimport { keys } from \"ramda\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * HTML and SVG attributes\r\n */\r\ntype Attributes =\r\n & JSXInternal.HTMLAttributes\r\n & JSXInternal.SVGAttributes\r\n & Record<string, any>\r\n\r\n/**\r\n * Child element\r\n */\r\ntype Child =\r\n | HTMLElement\r\n | SVGElement\r\n | Text\r\n | string\r\n | number\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Create an element\r\n *\r\n * @param tagName - HTML or SVG tag\r\n *\r\n * @return Element\r\n */\r\nfunction createElement(tagName: string): HTMLElement | SVGElement {\r\n switch (tagName) {\r\n\r\n /* SVG elements */\r\n case \"svg\":\r\n case \"path\":\r\n return document.createElementNS(\"http://www.w3.org/2000/svg\", tagName)\r\n\r\n /* HTML elements */\r\n default:\r\n return document.createElement(tagName)\r\n }\r\n}\r\n\r\n/**\r\n * Set an attribute\r\n *\r\n * @param el - Element\r\n * @param name - Attribute name\r\n * @param value - Attribute value\r\n */\r\nfunction setAttribute(\r\n el: HTMLElement | SVGElement, name: string, value: string) {\r\n switch (name) {\r\n\r\n /* Attributes to be ignored */\r\n case \"xmlns\":\r\n break\r\n\r\n /* Attributes of SVG elements */\r\n case \"viewBox\":\r\n case \"d\":\r\n if (typeof value !== \"boolean\")\r\n el.setAttributeNS(null, name, value)\r\n else if (value)\r\n el.setAttributeNS(null, name, \"\")\r\n break\r\n\r\n /* Attributes of HTML elements */\r\n default:\r\n if (typeof value !== \"boolean\")\r\n el.setAttribute(name, value)\r\n else if (value)\r\n el.setAttribute(name, \"\")\r\n }\r\n}\r\n\r\n/**\r\n * Append a child node to an element\r\n *\r\n * @param el - Element\r\n * @param child - Child node(s)\r\n */\r\nfunction appendChild(\r\n el: HTMLElement | SVGElement, child: Child | Child[]\r\n): void {\r\n\r\n /* Handle primitive types (including raw HTML) */\r\n if (typeof child === \"string\" || typeof child === \"number\") {\r\n el.innerHTML += child.toString()\r\n\r\n /* Handle nodes */\r\n } else if (child instanceof Node) {\r\n el.appendChild(child)\r\n\r\n /* Handle nested children */\r\n } else if (Array.isArray(child)) {\r\n for (const node of child)\r\n appendChild(el, node)\r\n }\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * JSX factory\r\n *\r\n * @param tagName - HTML or SVG tag\r\n * @param attributes - HTML attributes\r\n * @param children - Child elements\r\n *\r\n * @return Element\r\n */\r\nexport function h(\r\n tagName: string, attributes: Attributes | null, ...children: Child[]\r\n): HTMLElement | SVGElement {\r\n const el = createElement(tagName)\r\n\r\n /* Set attributes, if any */\r\n if (attributes)\r\n for (const attr of keys(attributes))\r\n setAttribute(el, attr, attributes[attr])\r\n\r\n /* Append child nodes */\r\n for (const child of children)\r\n appendChild(el, child)\r\n\r\n /* Return element */\r\n return el\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Namespace\r\n * ------------------------------------------------------------------------- */\r\n\r\nexport declare namespace h {\r\n namespace JSX {\r\n type Element = HTMLElement | SVGElement\r\n type IntrinsicElements = JSXInternal.IntrinsicElements\r\n }\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, defer, of } from \"rxjs\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Cache the last value emitted by an observable in session storage\r\n *\r\n * If the key is not found in session storage, the factory is executed and the\r\n * latest value emitted will automatically be persisted to sessions storage.\r\n * Note that the values emitted by the returned observable must be serializable\r\n * as `JSON`, or data will be lost.\r\n *\r\n * @template T - Value type\r\n *\r\n * @param key - Cache key\r\n * @param factory - Observable factory\r\n *\r\n * @return Value observable\r\n */\r\nexport function cache<T>(\r\n key: string, factory: () => Observable<T>\r\n): Observable<T> {\r\n return defer(() => {\r\n const data = sessionStorage.getItem(key)\r\n if (data) {\r\n return of(JSON.parse(data) as T)\r\n\r\n /* Retrieve value from observable factory and write to storage */\r\n } else {\r\n const value$ = factory()\r\n value$.subscribe(value => {\r\n try {\r\n sessionStorage.setItem(key, JSON.stringify(value))\r\n } catch (err) {\r\n /* Uncritical, just swallow */\r\n }\r\n })\r\n\r\n /* Return value */\r\n return value$\r\n }\r\n })\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { getElementOrThrow } from \"browser\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Translation keys\n */\ntype TranslateKey =\n | \"clipboard.copy\" /* Copy to clipboard */\n | \"clipboard.copied\" /* Copied to clipboard */\n | \"search.config.lang\" /* Search language */\n | \"search.config.pipeline\" /* Search pipeline */\n | \"search.config.separator\" /* Search separator */\n | \"search.result.placeholder\" /* Type to start searching */\n | \"search.result.none\" /* No matching documents */\n | \"search.result.one\" /* 1 matching document */\n | \"search.result.other\" /* # matching documents */\n\n/* ----------------------------------------------------------------------------\n * Data\n * ------------------------------------------------------------------------- */\n\n/**\n * Translations\n */\nlet lang: Record<string, string>\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Translate the given key\n *\n * @param key - Key to be translated\n * @param value - Value to be replaced\n *\n * @return Translation\n */\nexport function translate(key: TranslateKey, value?: string): string {\n if (typeof lang === \"undefined\") {\n const el = getElementOrThrow(\"#__lang\")\n lang = JSON.parse(el.textContent!)\n }\n if (typeof lang[key] === \"undefined\") {\n throw new ReferenceError(`Invalid translation: ${key}`)\n }\n return typeof value !== \"undefined\"\n ? lang[key].replace(\"#\", value)\n : lang[key]\n}\n\n/**\n * Truncate a string after the given number of characters\n *\n * This is not a very reasonable approach, since the summaries kind of suck.\n * It would be better to create something more intelligent, highlighting the\n * search occurrences and making a better summary out of it, but this note was\n * written three years ago, so who knows if we'll ever fix it.\n *\n * @param value - Value to be truncated\n * @param n - Number of characters\n *\n * @return Truncated value\n */\nexport function truncate(value: string, n: number): string {\n let i = n\n if (value.length > i) {\n while (value[i] !== \" \" && --i > 0); // tslint:disable-line\n return `${value.substring(0, i)}...`\n }\n return value\n}\n\n/**\n * Round a number for display with source facts\n *\n * This is a reverse engineered version of GitHub's weird rounding algorithm\n * for stars, forks and all other numbers. While all numbers below `1,000` are\n * returned as-is, bigger numbers are converted to fixed numbers:\n *\n * - `1,049` => `1k`\n * - `1,050` => `1.1k`\n * - `1,949` => `1.9k`\n * - `1,950` => `2k`\n *\n * @param value - Original value\n *\n * @return Rounded value\n */\nexport function round(value: number): string {\n if (value > 999) {\n const digits = +((value - 950) % 1000 > 99)\n return `${((value + 0.000001) / 1000).toFixed(digits)}k`\n } else {\n return value.toString()\n }\n}\n\n/**\n * Simple hash function\n *\n * @see https://bit.ly/2wsVjJ4 - Original source\n *\n * @param value - Value to be hashed\n *\n * @return Hash as 32bit integer\n */\nexport function hash(value: string): number {\n let h = 0\n for (let i = 0, len = value.length; i < len; i++) {\n h = ((h << 5) - h) + value.charCodeAt(i)\n h |= 0 // Convert to 32bit integer\n }\n return h\n }\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nexport * from \"./_\"\r\nexport * from \"./header\"\r\nexport * from \"./hero\"\r\nexport * from \"./main\"\r\nexport * from \"./navigation\"\r\nexport * from \"./search\"\r\nexport * from \"./shared\"\r\nexport * from \"./tabs\"\r\nexport * from \"./toc\"\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport * as ClipboardJS from \"clipboard\"\r\nimport { NEVER, Observable, Subject, fromEventPattern } from \"rxjs\"\r\nimport { mapTo, share, tap } from \"rxjs/operators\"\r\n\r\nimport { getElements } from \"browser\"\r\nimport { renderClipboardButton } from \"templates\"\r\nimport { translate } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Setup options\r\n */\r\ninterface SetupOptions {\r\n document$: Observable<Document> /* Document observable */\r\n dialog$: Subject<string> /* Dialog subject */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up clipboard\r\n *\r\n * This function implements the Clipboard.js integration and injects a button\r\n * into all code blocks when the document changes.\r\n *\r\n * @param options - Options\r\n *\r\n * @return Clipboard observable\r\n */\r\nexport function setupClipboard(\r\n { document$, dialog$ }: SetupOptions\r\n): Observable<ClipboardJS.Event> {\r\n if (!ClipboardJS.isSupported())\r\n return NEVER\r\n\r\n /* Inject 'copy-to-clipboard' buttons */\r\n document$.subscribe(() => {\r\n const blocks = getElements(\"pre > code\")\r\n blocks.forEach((block, index) => {\r\n const parent = block.parentElement!\r\n parent.id = `__code_${index}`\r\n parent.insertBefore(renderClipboardButton(parent.id), block)\r\n })\r\n })\r\n\r\n /* Initialize clipboard */\r\n const clipboard$ = fromEventPattern<ClipboardJS.Event>(next => {\r\n new ClipboardJS(\".md-clipboard\").on(\"success\", next)\r\n })\r\n .pipe(\r\n share()\r\n )\r\n\r\n /* Display notification for clipboard event */\r\n clipboard$\r\n .pipe(\r\n tap(ev => ev.clearSelection()),\r\n mapTo(translate(\"clipboard.copied\"))\r\n )\r\n .subscribe(dialog$)\r\n\r\n /* Return clipboard */\r\n return clipboard$\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Subject, animationFrameScheduler, of } from \"rxjs\"\r\nimport {\r\n delay,\r\n map,\r\n observeOn,\r\n switchMap,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport { createElement } from \"browser\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Setup options\r\n */\r\ninterface SetupOptions {\r\n duration?: number /* Display duration (default: 2s) */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up dialog\r\n *\r\n * @param options - Options\r\n *\r\n * @return Dialog observable\r\n */\r\nexport function setupDialog(\r\n { duration }: SetupOptions = {}\r\n): Subject<string> {\r\n const dialog$ = new Subject<string>()\r\n\r\n /* Create dialog */\r\n const dialog = createElement(\"div\") // TODO: improve scoping\r\n dialog.classList.add(\"md-dialog\", \"md-typeset\")\r\n\r\n /* Display dialog */\r\n dialog$\r\n .pipe(\r\n switchMap(text => of(document.body) // useComponent(\"container\")\r\n .pipe(\r\n map(container => container.appendChild(dialog)),\r\n observeOn(animationFrameScheduler),\r\n delay(1), // Strangley it doesnt work when we push things to the new animation frame...\r\n tap(el => {\r\n el.innerHTML = text\r\n el.setAttribute(\"data-md-state\", \"open\")\r\n }),\r\n delay(duration || 2000),\r\n tap(el => el.removeAttribute(\"data-md-state\")),\r\n delay(400),\r\n tap(el => {\r\n el.innerHTML = \"\"\r\n el.remove()\r\n })\r\n )\r\n )\r\n )\r\n .subscribe()\r\n\r\n /* Return dialog */\r\n return dialog$\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { NEVER, Observable, Subject, fromEvent, merge, of } from \"rxjs\"\nimport { ajax } from \"rxjs//ajax\"\nimport {\n bufferCount,\n catchError,\n debounceTime,\n distinctUntilChanged,\n distinctUntilKeyChanged,\n filter,\n map,\n pluck,\n sample,\n share,\n skip,\n switchMap,\n withLatestFrom\n} from \"rxjs/operators\"\n\nimport {\n Viewport,\n ViewportOffset,\n getElement,\n isAnchorLocation,\n isLocalLocation,\n replaceElement,\n setLocation,\n setLocationHash,\n setToggle,\n setViewportOffset\n} from \"browser\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * History state\n */\ninterface State {\n url: URL /* State URL */\n offset?: ViewportOffset /* State viewport offset */\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Setup options\n */\ninterface SetupOptions {\n document$: Subject<Document> /* Document subject */\n location$: Subject<URL> /* Location subject */\n viewport$: Observable<Viewport> /* Viewport observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Set up instant loading\n *\n * When fetching, theoretically, we could use `responseType: \"document\"`, but\n * since all MkDocs links are relative, we need to make sure that the current\n * location matches the document we just loaded. Otherwise any relative links\n * in the document could use the old location.\n *\n * This is the reason why we need to synchronize history events and the process\n * of fetching the document for navigation changes (except `popstate` events):\n *\n * 1. Fetch document via `XMLHTTPRequest`\n * 2. Set new location via `history.pushState`\n * 3. Parse and emit fetched document\n *\n * For `popstate` events, we must not use `history.pushState`, or the forward\n * history will be irreversibly overwritten. In case the request fails, the\n * location change is dispatched regularly.\n *\n * @param options - Options\n */\nexport function setupInstantLoading(\n urls: string[], { document$, viewport$, location$ }: SetupOptions\n): void {\n\n /* Disable automatic scroll restoration */\n if (\"scrollRestoration\" in history)\n history.scrollRestoration = \"manual\"\n\n /* Hack: ensure that reloads restore viewport offset */\n fromEvent(window, \"beforeunload\")\n .subscribe(() => {\n history.scrollRestoration = \"auto\"\n })\n\n /* Hack: ensure absolute favicon link to omit 404s on document switch */\n const favicon = getElement<HTMLLinkElement>(`link[rel=\"shortcut icon\"]`)\n if (typeof favicon !== \"undefined\")\n favicon.href = favicon.href // tslint:disable-line no-self-assignment\n\n /* Intercept link clicks and convert to state change */\n const state$ = fromEvent<MouseEvent>(document.body, \"click\")\n .pipe(\n filter(ev => !(ev.metaKey || ev.ctrlKey)),\n switchMap(ev => {\n if (ev.target instanceof HTMLElement) {\n const el = ev.target.closest(\"a\")\n if (\n el && !el.target &&\n isLocalLocation(el) &&\n urls.includes(el.href)\n ) {\n if (!isAnchorLocation(el))\n ev.preventDefault()\n return of(el)\n }\n }\n return NEVER\n }),\n map(el => ({ url: new URL(el.href) })),\n share<State>()\n )\n\n /* Always close search on link click */\n state$.subscribe(() => {\n setToggle(\"search\", false)\n })\n\n /* Filter state changes to dispatch */\n const push$ = state$\n .pipe(\n filter(({ url }) => !isAnchorLocation(url)),\n share()\n )\n\n /* Intercept popstate events (history back and forward) */\n const pop$ = fromEvent<PopStateEvent>(window, \"popstate\")\n .pipe(\n filter(ev => ev.state !== null),\n map(ev => ({\n url: new URL(location.href),\n offset: ev.state\n })),\n share<State>()\n )\n\n /* Emit location change */\n merge(push$, pop$)\n .pipe(\n distinctUntilChanged((prev, next) => prev.url.href === next.url.href),\n pluck(\"url\")\n )\n .subscribe(location$)\n\n /* Fetch document on location change */\n const ajax$ = location$\n .pipe(\n distinctUntilKeyChanged(\"pathname\"),\n skip(1),\n switchMap(url => ajax({\n url: url.href,\n responseType: \"text\",\n withCredentials: true\n })\n .pipe(\n catchError(() => {\n setLocation(url)\n return NEVER\n })\n )\n )\n )\n\n /* Set new location as soon as the document was fetched */\n push$\n .pipe(\n sample(ajax$)\n )\n .subscribe(({ url }) => {\n history.pushState({}, \"\", url.toString())\n })\n\n /* Parse and emit document */\n const dom = new DOMParser()\n ajax$\n .pipe(\n map(({ response }) => dom.parseFromString(response, \"text/html\"))\n )\n .subscribe(document$)\n\n /* Intercept instant loading */\n const instant$ = merge(push$, pop$)\n .pipe(\n sample(document$)\n )\n\n // TODO: this must be combined with search scroll restoration on mobile\n instant$.subscribe(({ url, offset }) => {\n if (url.hash && !offset) {\n setLocationHash(url.hash)\n } else {\n setViewportOffset(offset || { y: 0 })\n }\n })\n\n /* Replace document metadata */\n instant$\n .pipe(\n withLatestFrom(document$)\n )\n .subscribe(([, { title, head }]) => {\n document.dispatchEvent(new CustomEvent(\"DOMContentSwitch\"))\n document.title = title\n\n /* Replace meta tags */\n for (const selector of [\n `link[rel=\"canonical\"]`,\n `meta[name=\"author\"]`,\n `meta[name=\"description\"]`\n ]) {\n const next = getElement(selector, head)\n const prev = getElement(selector, document.head)\n if (\n typeof next !== \"undefined\" &&\n typeof prev !== \"undefined\"\n ) {\n replaceElement(prev, next)\n }\n }\n })\n\n /* Debounce update of viewport offset */\n viewport$\n .pipe(\n debounceTime(250),\n distinctUntilKeyChanged(\"offset\")\n )\n .subscribe(({ offset }) => {\n history.replaceState(offset, \"\")\n })\n\n /* Set viewport offset from history */\n merge(state$, pop$)\n .pipe(\n bufferCount(2, 1),\n filter(([prev, next]) => {\n return prev.url.pathname === next.url.pathname\n && !isAnchorLocation(next.url)\n }),\n map(([, state]) => state)\n )\n .subscribe(({ offset }) => {\n setViewportOffset(offset || { y: 0 })\n })\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable } from \"rxjs\"\r\nimport {\r\n filter,\r\n map,\r\n share,\r\n withLatestFrom\r\n} from \"rxjs/operators\"\r\n\r\nimport {\r\n Key,\r\n getActiveElement,\r\n getElement,\r\n getElements,\r\n getToggle,\r\n isSusceptibleToKeyboard,\r\n setElementFocus,\r\n setElementSelection,\r\n setToggle,\r\n watchKeyboard\r\n} from \"browser\"\r\nimport { useComponent } from \"components\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Keyboard mode\r\n */\r\nexport type KeyboardMode =\r\n | \"global\" /* Global */\r\n | \"search\" /* Search is open */\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Keyboard\r\n */\r\nexport interface Keyboard extends Key {\r\n mode: KeyboardMode /* Keyboard mode */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up keyboard\r\n *\r\n * This function will set up the keyboard handlers and ensure that keys are\r\n * correctly propagated. Currently there are two modes:\r\n *\r\n * - `global`: This mode is active when the search is closed. It is intended\r\n * to assign hotkeys to specific functions of the site. Currently the search,\r\n * previous and next page can be triggered.\r\n *\r\n * - `search`: This mode is active when the search is open. It maps certain\r\n * navigational keys to offer search results that can be entirely navigated\r\n * through keyboard input.\r\n *\r\n * The keyboard observable is returned and can be used to monitor the keyboard\r\n * in order toassign further hotkeys to custom functions.\r\n *\r\n * @return Keyboard observable\r\n */\r\nexport function setupKeyboard(): Observable<Keyboard> {\r\n const keyboard$ = watchKeyboard()\r\n .pipe(\r\n map<Key, Keyboard>(key => ({\r\n mode: getToggle(\"search\") ? \"search\" : \"global\",\r\n ...key\r\n })),\r\n filter(({ mode }) => {\r\n if (mode === \"global\") {\r\n const active = getActiveElement()\r\n if (typeof active !== \"undefined\")\r\n return !isSusceptibleToKeyboard(active)\r\n }\r\n return true\r\n }),\r\n share()\r\n )\r\n\r\n /* Set up search keyboard handlers */\r\n keyboard$\r\n .pipe(\r\n filter(({ mode }) => mode === \"search\"),\r\n withLatestFrom(\r\n useComponent(\"search-query\"),\r\n useComponent(\"search-result\")\r\n )\r\n )\r\n .subscribe(([key, query, result]) => {\r\n const active = getActiveElement()\r\n switch (key.type) {\r\n\r\n /* Enter: prevent form submission */\r\n case \"Enter\":\r\n if (active === query)\r\n key.claim()\r\n break\r\n\r\n /* Escape or Tab: close search */\r\n case \"Escape\":\r\n case \"Tab\":\r\n setToggle(\"search\", false)\r\n setElementFocus(query, false)\r\n break\r\n\r\n /* Vertical arrows: select previous or next search result */\r\n case \"ArrowUp\":\r\n case \"ArrowDown\":\r\n if (typeof active === \"undefined\") {\r\n setElementFocus(query)\r\n } else {\r\n const els = [query, ...getElements(\"[href]\", result)]\r\n const i = Math.max(0, (\r\n Math.max(0, els.indexOf(active)) + els.length + (\r\n key.type === \"ArrowUp\" ? -1 : +1\r\n )\r\n ) % els.length)\r\n setElementFocus(els[i])\r\n }\r\n\r\n /* Prevent scrolling of page */\r\n key.claim()\r\n break\r\n\r\n /* All other keys: hand to search query */\r\n default:\r\n if (query !== getActiveElement())\r\n setElementFocus(query)\r\n }\r\n })\r\n\r\n /* Set up global keyboard handlers */\r\n keyboard$\r\n .pipe(\r\n filter(({ mode }) => mode === \"global\"),\r\n withLatestFrom(useComponent(\"search-query\"))\r\n )\r\n .subscribe(([key, query]) => {\r\n switch (key.type) {\r\n\r\n /* Open search and select query */\r\n case \"f\":\r\n case \"s\":\r\n case \"/\":\r\n setElementFocus(query)\r\n setElementSelection(query)\r\n key.claim()\r\n break\r\n\r\n /* Go to previous page */\r\n case \"p\":\r\n case \",\":\r\n const prev = getElement(\"[href][rel=prev]\")\r\n if (typeof prev !== \"undefined\")\r\n prev.click()\r\n break\r\n\r\n /* Go to next page */\r\n case \"n\":\r\n case \".\":\r\n const next = getElement(\"[href][rel=next]\")\r\n if (typeof next !== \"undefined\")\r\n next.click()\r\n break\r\n }\r\n })\r\n\r\n /* Return keyboard */\r\n return keyboard$\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { EMPTY, Observable, of } from \"rxjs\"\r\nimport {\r\n distinctUntilChanged,\r\n map,\r\n scan,\r\n shareReplay,\r\n switchMap\r\n} from \"rxjs/operators\"\r\n\r\nimport { getElement, replaceElement } from \"browser\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Component\r\n */\r\nexport type Component =\r\n | \"announce\" /* Announcement bar */\r\n | \"container\" /* Container */\r\n | \"header\" /* Header */\r\n | \"header-title\" /* Header title */\r\n | \"hero\" /* Hero */\r\n | \"main\" /* Main area */\r\n | \"navigation\" /* Navigation */\r\n | \"search\" /* Search */\r\n | \"search-query\" /* Search input */\r\n | \"search-reset\" /* Search reset */\r\n | \"search-result\" /* Search results */\r\n | \"skip\" /* Skip link */\r\n | \"tabs\" /* Tabs */\r\n | \"toc\" /* Table of contents */\r\n\r\n/**\r\n * Component map\r\n */\r\nexport type ComponentMap = {\r\n [P in Component]?: HTMLElement\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch options\r\n */\r\ninterface WatchOptions {\r\n document$: Observable<Document> /* Document observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Component map observable\r\n */\r\nlet components$: Observable<ComponentMap>\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up bindings to components with given names\r\n *\r\n * This function will maintain bindings to the elements identified by the given\r\n * names in-between document switches and update the elements in-place.\r\n *\r\n * @param names - Component names\r\n * @param options - Options\r\n */\r\nexport function setupComponents(\r\n names: Component[], { document$ }: WatchOptions\r\n): void {\r\n components$ = document$\r\n .pipe(\r\n\r\n /* Build component map */\r\n map(document => names.reduce<ComponentMap>((components, name) => {\r\n const el = getElement(`[data-md-component=${name}]`, document)\r\n return {\r\n ...components,\r\n ...typeof el !== \"undefined\" ? { [name]: el } : {}\r\n }\r\n }, {})),\r\n\r\n /* Re-compute component map on document switch */\r\n scan((prev, next) => {\r\n for (const name of names) {\r\n switch (name) {\r\n\r\n /* Top-level components: update */\r\n case \"announce\":\r\n case \"header-title\":\r\n case \"container\":\r\n case \"skip\":\r\n if (name in prev && typeof prev[name] !== \"undefined\") {\r\n replaceElement(prev[name]!, next[name]!)\r\n prev[name] = next[name]\r\n }\r\n break\r\n\r\n /* All other components: rebind */\r\n default:\r\n if (typeof next[name] !== \"undefined\")\r\n prev[name] = getElement(`[data-md-component=${name}]`)\r\n else\r\n delete prev[name]\r\n }\r\n }\r\n return prev\r\n }),\r\n\r\n /* Convert to hot observable */\r\n shareReplay(1)\r\n )\r\n}\r\n\r\n/**\r\n * Retrieve a component\r\n *\r\n * The returned observable will only re-emit if the element changed, i.e. if\r\n * it was replaced from a document which was switched to.\r\n *\r\n * @template T - Element type\r\n *\r\n * @param name - Component name\r\n *\r\n * @return Component observable\r\n */\r\nexport function useComponent<T extends HTMLInputElement>(\r\n name: \"search-query\"\r\n): Observable<T>\r\nexport function useComponent<T extends HTMLElement>(\r\n name: Component\r\n): Observable<T>\r\nexport function useComponent<T extends HTMLElement>(\r\n name: Component\r\n): Observable<T> {\r\n return components$\r\n .pipe(\r\n switchMap(components => (\r\n typeof components[name] !== \"undefined\"\r\n ? of(components[name] as T)\r\n : EMPTY\r\n )),\r\n distinctUntilChanged()\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set anchor blur\r\n *\r\n * @param el - Anchor element\r\n * @param value - Whether the anchor is blurred\r\n */\r\nexport function setAnchorBlur(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.setAttribute(\"data-md-state\", value ? \"blur\" : \"\")\r\n}\r\n\r\n/**\r\n * Reset anchor blur\r\n *\r\n * @param el - Anchor element\r\n */\r\nexport function resetAnchorBlur(\r\n el: HTMLElement\r\n): void {\r\n el.removeAttribute(\"data-md-state\")\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set anchor active\r\n *\r\n * @param el - Anchor element\r\n * @param value - Whether the anchor is active\r\n */\r\nexport function setAnchorActive(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.classList.toggle(\"md-nav__link--active\", value)\r\n}\r\n\r\n/**\r\n * Reset anchor active\r\n *\r\n * @param el - Anchor element\r\n */\r\nexport function resetAnchorActive(\r\n el: HTMLElement\r\n): void {\r\n el.classList.remove(\"md-nav__link--active\")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nexport * from \"./sidebar\"\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { h, translate } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * CSS classes\r\n */\r\nconst css = {\r\n container: \"md-clipboard md-icon\"\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Path of `file-search-outline` icon\r\n */\r\nconst path =\r\n \"M19,21H8V7H19M19,5H8A2,2 0 0,0 6,7V21A2,2 0 0,0 8,23H19A2,2 0 0,0 \" +\r\n \"21,21V7A2,2 0 0,0 19,5M16,1H4A2,2 0 0,0 2,3V17H4V3H16V1Z\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Render a 'copy-to-clipboard' button\r\n *\r\n * @param id - Unique identifier\r\n *\r\n * @return Element\r\n */\r\nexport function renderClipboardButton(\r\n id: string\r\n) {\r\n return (\r\n <button\r\n class={css.container}\r\n title={translate(\"clipboard.copy\")}\r\n data-clipboard-target={`#${id} > code`}\r\n >\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 24 24\">\r\n <path d={path}></path>\r\n </svg>\r\n </button>\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { SearchResult } from \"integrations/search\"\r\nimport { h, truncate } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * CSS classes\r\n */\r\nconst css = {\r\n item: \"md-search-result__item\",\r\n link: \"md-search-result__link\",\r\n article: \"md-search-result__article md-search-result__article--document\",\r\n section: \"md-search-result__article\",\r\n title: \"md-search-result__title\",\r\n teaser: \"md-search-result__teaser\"\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Path of `content-copy` icon\r\n */\r\nconst path =\r\n \"M14,2H6A2,2 0 0,0 4,4V20A2,2 0 0,0 6,22H13C12.59,21.75 12.2,21.44 \" +\r\n \"11.86,21.1C11.53,20.77 11.25,20.4 11,20H6V4H13V9H18V10.18C18.71,10.34 \" +\r\n \"19.39,10.61 20,11V8L14,2M20.31,18.9C21.64,16.79 21,14 \" +\r\n \"18.91,12.68C16.8,11.35 14,12 12.69,14.08C11.35,16.19 12,18.97 \" +\r\n \"14.09,20.3C15.55,21.23 17.41,21.23 \" +\r\n \"18.88,20.32L22,23.39L23.39,22L20.31,18.9M16.5,19A2.5,2.5 0 0,1 \" +\r\n \"14,16.5A2.5,2.5 0 0,1 16.5,14A2.5,2.5 0 0,1 19,16.5A2.5,2.5 0 0,1 16.5,19Z\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Render a search result\r\n *\r\n * @param result - Search result\r\n *\r\n * @return Element\r\n */\r\nexport function renderSearchResult(\r\n { article, sections }: SearchResult\r\n) {\r\n\r\n /* Render icon */\r\n const icon = (\r\n <div class=\"md-search-result__icon md-icon\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 24 24\">\r\n <path d={path}></path>\r\n </svg>\r\n </div>\r\n )\r\n\r\n /* Render article and sections */\r\n const children = [article, ...sections].map(document => {\r\n const { location, title, text } = document\r\n return (\r\n <a href={location} class={css.link} tabIndex={-1}>\r\n <article class={\"parent\" in document ? css.section : css.article}>\r\n {!(\"parent\" in document) && icon}\r\n <h1 class={css.title}>{title}</h1>\r\n {text.length > 0 && <p class={css.teaser}>{truncate(text, 320)}</p>}\r\n </article>\r\n </a>\r\n )\r\n })\r\n\r\n /* Render search result */\r\n return (\r\n <li class={css.item}>\r\n {children}\r\n </li>\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { SourceFacts } from \"patches/source\"\r\nimport { h } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * CSS classes\r\n */\r\nconst css = {\r\n facts: \"md-source__facts\",\r\n fact: \"md-source__fact\"\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Render source facts\r\n *\r\n * @param facts - Source facts\r\n *\r\n * @return Element\r\n */\r\nexport function renderSource(\r\n facts: SourceFacts\r\n) {\r\n const children = facts.map(fact => (\r\n <li class={css.fact}>{fact}</li>\r\n ))\r\n return (\r\n <ul class={css.facts}>\r\n {children}\r\n </ul>\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { h } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Data\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * CSS classes\r\n */\r\nconst css = {\r\n wrapper: \"md-typeset__scrollwrap\",\r\n table: \"md-typeset__table\"\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Render a table inside a wrapper to improve scrolling on mobile\r\n *\r\n * @param table - Table element\r\n *\r\n * @return Element\r\n */\r\nexport function renderTable(\r\n table: HTMLTableElement\r\n) {\r\n return (\r\n <div class={css.wrapper}>\r\n <div class={css.table}>\r\n {table}\r\n </div>\r\n </div>\r\n )\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Set sidebar offset\n *\n * @param el - Sidebar element\n * @param value - Sidebar offset\n */\nexport function setSidebarOffset(\n el: HTMLElement, value: number\n): void {\n el.style.top = `${value}px`\n}\n\n/**\n * Reset sidebar offset\n *\n * @param el - Sidebar element\n */\nexport function resetSidebarOffset(\n el: HTMLElement\n): void {\n el.style.top = \"\"\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Set sidebar height\n *\n * @param el - Sidebar element\n * @param value - Sidebar height\n */\nexport function setSidebarHeight(\n el: HTMLElement, value: number\n): void {\n el.style.height = `${value}px`\n}\n\n/**\n * Reset sidebar height\n *\n * @param el - Sidebar element\n */\nexport function resetSidebarHeight(\n el: HTMLElement\n): void {\n el.style.height = \"\"\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nexport * from \"./_\"\r\nexport * from \"./react\"\r\nexport * from \"./set\"\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport {\n ArticleDocument,\n SearchDocumentMap,\n SectionDocument,\n setupSearchDocumentMap\n} from \"../document\"\nimport {\n SearchHighlightFactoryFn,\n setupSearchHighlighter\n} from \"../highlighter\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Search index configuration\n */\nexport interface SearchIndexConfig {\n lang: string[] /* Search languages */\n separator: string /* Search separator */\n}\n\n/**\n * Search index document\n */\nexport interface SearchIndexDocument {\n location: string /* Document location */\n title: string /* Document title */\n text: string /* Document text */\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Search index pipeline function\n */\nexport type SearchIndexPipelineFn =\n | \"stemmer\" /* Stemmer */\n | \"stopWordFilter\" /* Stop word filter */\n | \"trimmer\" /* Trimmer */\n\n/**\n * Search index pipeline\n */\nexport type SearchIndexPipeline = SearchIndexPipelineFn[]\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Search index\n *\n * This interfaces describes the format of the `search_index.json` file which\n * is automatically built by the MkDocs search plugin.\n */\nexport interface SearchIndex {\n config: SearchIndexConfig /* Search index configuration */\n docs: SearchIndexDocument[] /* Search index documents */\n index?: object | string /* Prebuilt or serialized index */\n pipeline?: SearchIndexPipeline /* Search index pipeline */\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Search result\n */\nexport interface SearchResult {\n article: ArticleDocument /* Article document */\n sections: SectionDocument[] /* Section documents */\n}\n\n/* ----------------------------------------------------------------------------\n * Class\n * ------------------------------------------------------------------------- */\n\n/**\n * Search\n *\n * Note that `lunr` is injected via Webpack, as it will otherwise also be\n * bundled in the application bundle.\n */\nexport class Search {\n\n /**\n * Search document mapping\n *\n * A mapping of URLs (including hash fragments) to the actual articles and\n * sections of the documentation. The search document mapping must be created\n * regardless of whether the index was prebuilt or not, as `lunr` itself will\n * only store the actual index.\n */\n protected documents: SearchDocumentMap\n\n /**\n * Search highlight factory function\n */\n protected highlight: SearchHighlightFactoryFn\n\n /**\n * The `lunr` search index\n */\n protected index: lunr.Index\n\n /**\n * Create the search integration\n *\n * @param data - Search index\n */\n public constructor({ config, docs, pipeline, index }: SearchIndex) {\n this.documents = setupSearchDocumentMap(docs)\n this.highlight = setupSearchHighlighter(config)\n\n /* If no index was given, create it */\n if (typeof index === \"undefined\") {\n this.index = lunr(function() {\n pipeline = pipeline || [\"trimmer\", \"stopWordFilter\"]\n\n /* Set up pipeline according to configuration */\n this.pipeline.reset()\n for (const fn of pipeline)\n this.pipeline.add(lunr[fn])\n\n /* Set up alternate search languages */\n if (config.lang.length === 1 && config.lang[0] !== \"en\") {\n this.use((lunr as any)[config.lang[0]])\n } else if (config.lang.length > 1) {\n this.use((lunr as any).multiLanguage(...config.lang))\n }\n\n /* Set up fields and reference */\n this.field(\"title\", { boost: 1000 })\n this.field(\"text\")\n this.ref(\"location\")\n\n /* Index documents */\n for (const doc of docs)\n this.add(doc)\n })\n\n /* Prebuilt or serialized index */\n } else {\n this.index = lunr.Index.load(\n typeof index === \"string\"\n ? JSON.parse(index)\n : index\n )\n }\n }\n\n /**\n * Search for matching documents\n *\n * The search index which MkDocs provides is divided up into articles, which\n * contain the whole content of the individual pages, and sections, which only\n * contain the contents of the subsections obtained by breaking the individual\n * pages up at `h1` ... `h6`. As there may be many sections on different pages\n * with identical titles (for example within this very project, e.g. \"Usage\"\n * or \"Installation\"), they need to be put into the context of the containing\n * page. For this reason, section results are grouped within their respective\n * articles which are the top-level results that are returned.\n *\n * @param value - Query value\n *\n * @return Search results\n */\n public query(value: string): SearchResult[] {\n if (value) {\n try {\n\n /* Group sections by containing article */\n const groups = this.index.search(value)\n .reduce((results, result) => {\n const document = this.documents.get(result.ref)\n if (typeof document !== \"undefined\") {\n if (\"parent\" in document) {\n const ref = document.parent.location\n results.set(ref, [...results.get(ref) || [], result])\n } else {\n const ref = document.location\n results.set(ref, results.get(ref) || [])\n }\n }\n return results\n }, new Map<string, lunr.Index.Result[]>())\n\n /* Create highlighter for query */\n const fn = this.highlight(value)\n\n /* Map groups to search documents */\n return [...groups].map(([ref, sections]) => ({\n article: fn(this.documents.get(ref) as ArticleDocument),\n sections: sections.map(section => {\n return fn(this.documents.get(section.ref) as SectionDocument)\n })\n }))\n\n /* Log errors to console (for now) */\n } catch (err) {\n // tslint:disable-next-line no-console\n console.warn(`Invalid query: ${value} – see https://bit.ly/2s3ChXG`)\n }\n }\n\n /* Return nothing in case of error or empty query */\n return []\n }\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport * as escapeHTML from \"escape-html\"\r\n\r\nimport { SearchIndexDocument } from \"../_\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * A top-level article\r\n */\r\nexport interface ArticleDocument extends SearchIndexDocument {\r\n linked: boolean /* Whether the section was linked */\r\n}\r\n\r\n/**\r\n * A section of an article\r\n */\r\nexport interface SectionDocument extends SearchIndexDocument {\r\n parent: ArticleDocument /* Parent article */\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Search document\r\n */\r\nexport type SearchDocument =\r\n | ArticleDocument\r\n | SectionDocument\r\n\r\n/**\r\n * Search document mapping\r\n */\r\nexport type SearchDocumentMap = Map<string, SearchDocument>\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Create a search document mapping\r\n *\r\n * @param docs - Search index documents\r\n *\r\n * @return Search document map\r\n */\r\nexport function setupSearchDocumentMap(\r\n docs: SearchIndexDocument[]\r\n): SearchDocumentMap {\r\n const documents = new Map<string, SearchDocument>()\r\n for (const doc of docs) {\r\n const [path, hash] = doc.location.split(\"#\")\r\n\r\n /* Extract location and title */\r\n const location = doc.location\r\n const title = doc.title\r\n\r\n /* Escape and cleanup text */\r\n const text = escapeHTML(doc.text)\r\n .replace(/\\s+(?=[,.:;!?])/g, \"\")\r\n .replace(/\\s+/g, \" \")\r\n\r\n /* Handle section */\r\n if (hash) {\r\n const parent = documents.get(path) as ArticleDocument\r\n\r\n /* Ignore first section, override article */\r\n if (!parent.linked) {\r\n parent.title = doc.title\r\n parent.text = text\r\n parent.linked = true\r\n\r\n /* Add subsequent section */\r\n } else {\r\n documents.set(location, {\r\n location,\r\n title,\r\n text,\r\n parent\r\n })\r\n }\r\n\r\n /* Add article */\r\n } else {\r\n documents.set(location, {\r\n location,\r\n title,\r\n text,\r\n linked: false\r\n })\r\n }\r\n }\r\n return documents\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { SearchIndexConfig } from \"../_\"\r\nimport { SearchDocument } from \"../document\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Search highlight function\r\n *\r\n * @template T - Search document type\r\n *\r\n * @param document - Search document\r\n *\r\n * @return Highlighted document\r\n */\r\nexport type SearchHighlightFn = <\r\n T extends SearchDocument\r\n>(document: Readonly<T>) => T\r\n\r\n/**\r\n * Search highlight factory function\r\n *\r\n * @param value - Query value\r\n *\r\n * @return Search highlight function\r\n */\r\nexport type SearchHighlightFactoryFn = (value: string) => SearchHighlightFn\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Create a search highlighter\r\n *\r\n * @param config - Search index configuration\r\n *\r\n * @return Search highlight factory function\r\n */\r\nexport function setupSearchHighlighter(\r\n config: SearchIndexConfig\r\n): SearchHighlightFactoryFn {\r\n const separator = new RegExp(config.separator, \"img\")\r\n const highlight = (_: unknown, data: string, term: string) => {\r\n return `${data}<em>${term}</em>`\r\n }\r\n\r\n /* Return factory function */\r\n return (value: string) => {\r\n value = value\r\n .replace(/[\\s*+-:~^]+/g, \" \")\r\n .trim()\r\n\r\n /* Create search term match expression */\r\n const match = new RegExp(`(^|${config.separator})(${\r\n value\r\n .replace(/[|\\\\{}()[\\]^$+*?.-]/g, \"\\\\$&\")\r\n .replace(separator, \"|\")\r\n })`, \"img\")\r\n\r\n /* Highlight document */\r\n return document => ({\r\n ...document,\r\n title: document.title.replace(match, highlight),\r\n text: document.text.replace(match, highlight)\r\n })\r\n }\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Search transformation function\r\n *\r\n * @param value - Query value\r\n *\r\n * @return Transformed query value\r\n */\r\nexport type SearchTransformFn = (value: string) => string\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Default transformation function\r\n *\r\n * Rogue control characters are filtered before handing the query to the\r\n * search index, as `lunr` will throw otherwise.\r\n *\r\n * @param value - Query value\r\n *\r\n * @return Transformed query value\r\n */\r\nexport function defaultTransform(value: string): string {\r\n return value\r\n .replace(/(?:^|\\s+)[*+-:^~]+(?=\\s+|$)/g, \"\")\r\n .trim()\r\n .replace(/\\s+|\\b$/g, \"* \")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A RTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { SearchIndex, SearchResult } from \"../../_\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Search message type\r\n */\r\nexport const enum SearchMessageType {\r\n SETUP, /* Search index setup */\r\n READY, /* Search index ready */\r\n QUERY, /* Search query */\r\n RESULT /* Search results */\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * A message containing the data necessary to setup the search index\r\n */\r\nexport interface SearchSetupMessage {\r\n type: SearchMessageType.SETUP /* Message type */\r\n data: SearchIndex /* Message data */\r\n}\r\n\r\n/**\r\n * A message indicating the search index is ready\r\n */\r\nexport interface SearchReadyMessage {\r\n type: SearchMessageType.READY /* Message type */\r\n}\r\n\r\n/**\r\n * A message containing a search query\r\n */\r\nexport interface SearchQueryMessage {\r\n type: SearchMessageType.QUERY /* Message type */\r\n data: string /* Message data */\r\n}\r\n\r\n/**\r\n * A message containing results for a search query\r\n */\r\nexport interface SearchResultMessage {\r\n type: SearchMessageType.RESULT /* Message type */\r\n data: SearchResult[] /* Message data */\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * A message exchanged with the search worker\r\n */\r\nexport type SearchMessage =\r\n | SearchSetupMessage\r\n | SearchReadyMessage\r\n | SearchQueryMessage\r\n | SearchResultMessage\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Type guard for search setup messages\r\n *\r\n * @param message - Search worker message\r\n *\r\n * @return Test result\r\n */\r\nexport function isSearchSetupMessage(\r\n message: SearchMessage\r\n): message is SearchSetupMessage {\r\n return message.type === SearchMessageType.SETUP\r\n}\r\n\r\n/**\r\n * Type guard for search ready messages\r\n *\r\n * @param message - Search worker message\r\n *\r\n * @return Test result\r\n */\r\nexport function isSearchReadyMessage(\r\n message: SearchMessage\r\n): message is SearchReadyMessage {\r\n return message.type === SearchMessageType.READY\r\n}\r\n\r\n/**\r\n * Type guard for search query messages\r\n *\r\n * @param message - Search worker message\r\n *\r\n * @return Test result\r\n */\r\nexport function isSearchQueryMessage(\r\n message: SearchMessage\r\n): message is SearchQueryMessage {\r\n return message.type === SearchMessageType.QUERY\r\n}\r\n\r\n/**\r\n * Type guard for search result messages\r\n *\r\n * @param message - Search worker message\r\n *\r\n * @return Test result\r\n */\r\nexport function isSearchResultMessage(\r\n message: SearchMessage\r\n): message is SearchResultMessage {\r\n return message.type === SearchMessageType.RESULT\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A RTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { identity } from \"ramda\"\r\nimport { Observable, Subject, asyncScheduler } from \"rxjs\"\r\nimport {\r\n map,\r\n observeOn,\r\n shareReplay,\r\n withLatestFrom\r\n} from \"rxjs/operators\"\r\n\r\nimport { WorkerHandler, watchWorker } from \"browser\"\r\nimport { translate } from \"utilities\"\r\n\r\nimport { SearchIndex, SearchIndexPipeline } from \"../../_\"\r\nimport {\r\n SearchMessage,\r\n SearchMessageType,\r\n SearchSetupMessage,\r\n isSearchResultMessage\r\n} from \"../message\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Setup options\r\n */\r\ninterface SetupOptions {\r\n index$: Observable<SearchIndex> /* Search index observable */\r\n base$: Observable<string> /* Location base observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up search index\r\n *\r\n * @param data - Search index\r\n *\r\n * @return Search index\r\n */\r\nfunction setupSearchIndex(\r\n { config, docs, index }: SearchIndex\r\n): SearchIndex {\r\n\r\n /* Override default language with value from translation */\r\n if (config.lang.length === 1 && config.lang[0] === \"en\")\r\n config.lang = [translate(\"search.config.lang\")]\r\n\r\n /* Override default separator with value from translation */\r\n if (config.separator === \"[\\s\\-]+\")\r\n config.separator = translate(\"search.config.separator\")\r\n\r\n /* Set pipeline from translation */\r\n const pipeline = translate(\"search.config.pipeline\")\r\n .split(/\\s*,\\s*/)\r\n .filter(identity) as SearchIndexPipeline\r\n\r\n /* Return search index after defaulting */\r\n return { config, docs, index, pipeline }\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set up search web worker\r\n *\r\n * This function will create a web worker to set up and query the search index\r\n * which is done using `lunr`. The index must be passed as an observable to\r\n * enable hacks like _localsearch_ via search index embedding as JSON.\r\n *\r\n * @param url - Worker URL\r\n * @param options - Options\r\n *\r\n * @return Worker handler\r\n */\r\nexport function setupSearchWorker(\r\n url: string, { index$, base$ }: SetupOptions\r\n): WorkerHandler<SearchMessage> {\r\n const worker = new Worker(url)\r\n\r\n /* Create communication channels and resolve relative links */\r\n const tx$ = new Subject<SearchMessage>()\r\n const rx$ = watchWorker(worker, { tx$ })\r\n .pipe(\r\n withLatestFrom(base$),\r\n map(([message, base]) => {\r\n if (isSearchResultMessage(message)) {\r\n for (const { article, sections } of message.data) {\r\n article.location = `${base}/${article.location}`\r\n for (const section of sections)\r\n section.location = `${base}/${section.location}`\r\n }\r\n }\r\n return message\r\n }),\r\n shareReplay(1)\r\n )\r\n\r\n /* Set up search index */\r\n index$\r\n .pipe(\r\n map<SearchIndex, SearchSetupMessage>(index => ({\r\n type: SearchMessageType.SETUP,\r\n data: setupSearchIndex(index)\r\n })),\r\n observeOn(asyncScheduler)\r\n )\r\n .subscribe(tx$.next.bind(tx$))\r\n\r\n /* Return worker handler */\r\n return { tx$, rx$ }\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nexport * from \"./_\"\r\nexport * from \"./react\"\r\nexport * from \"./set\"\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport {\n MonoTypeOperatorFunction,\n Observable,\n animationFrameScheduler,\n combineLatest,\n pipe\n} from \"rxjs\"\nimport {\n distinctUntilChanged,\n finalize,\n map,\n observeOn,\n tap,\n withLatestFrom\n} from \"rxjs/operators\"\n\nimport { Viewport } from \"browser\"\n\nimport { Header } from \"../../../header\"\nimport { Main } from \"../../../main\"\nimport { Sidebar } from \"../_\"\nimport {\n resetSidebarHeight,\n resetSidebarOffset,\n setSidebarHeight,\n setSidebarOffset\n} from \"../set\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch options\n */\ninterface WatchOptions {\n main$: Observable<Main> /* Main area observable */\n viewport$: Observable<Viewport> /* Viewport observable */\n}\n\n/**\n * Apply options\n */\ninterface ApplyOptions {\n header$: Observable<Header> /* Header observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch sidebar\n *\n * This function returns an observable that computes the visual parameters of\n * the sidebar which depends on the vertical viewport offset, as well as the\n * height of the main area. When the page is scrolled beyond the header, the\n * sidebar is locked and fills the remaining space.\n *\n * @param el - Sidebar element\n * @param options - Options\n *\n * @return Sidebar observable\n */\nexport function watchSidebar(\n el: HTMLElement, { main$, viewport$ }: WatchOptions\n): Observable<Sidebar> {\n const adjust = el.parentElement!.offsetTop\n - el.parentElement!.parentElement!.offsetTop\n\n /* Compute the sidebar's available height and if it should be locked */\n return combineLatest([main$, viewport$])\n .pipe(\n map(([{ offset, height }, { offset: { y } }]) => {\n height = height\n + Math.min(adjust, Math.max(0, y - offset))\n - adjust\n return {\n height,\n lock: y >= offset + adjust\n }\n }),\n distinctUntilChanged<Sidebar>((a, b) => {\n return a.height === b.height\n && a.lock === b.lock\n })\n )\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Apply sidebar\n *\n * @param el - Sidebar element\n * @param options - Options\n *\n * @return Operator function\n */\nexport function applySidebar(\n el: HTMLElement, { header$ }: ApplyOptions\n): MonoTypeOperatorFunction<Sidebar> {\n return pipe(\n\n /* Defer repaint to next animation frame */\n observeOn(animationFrameScheduler),\n withLatestFrom(header$),\n tap(([{ height, lock }, { height: offset }]) => {\n setSidebarHeight(el, height)\n\n /* Set offset in locked state depending on header height */\n if (lock)\n setSidebarOffset(el, offset)\n else\n resetSidebarOffset(el)\n }),\n\n /* Re-map to sidebar */\n map(([sidebar]) => sidebar),\n\n /* Reset on complete or error */\n finalize(() => {\n resetSidebarOffset(el)\n resetSidebarHeight(el)\n })\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nexport * from \"./_\"\r\nexport * from \"./anchor\"\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport {\n Observable,\n OperatorFunction,\n combineLatest,\n of,\n pipe\n} from \"rxjs\"\nimport { map, switchMap } from \"rxjs/operators\"\n\nimport { Viewport, getElements } from \"browser\"\n\nimport { Header } from \"../../header\"\nimport { Main } from \"../../main\"\nimport {\n Sidebar,\n applySidebar,\n watchSidebar\n} from \"../../shared\"\nimport {\n AnchorList,\n applyAnchorList,\n watchAnchorList\n} from \"../anchor\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Table of contents for [tablet -]\n */\ninterface TableOfContentsBelowTablet {} // tslint:disable-line\n\n/**\n * Table of contents for [tablet +]\n */\ninterface TableOfContentsAboveTablet {\n sidebar: Sidebar /* Sidebar */\n anchors: AnchorList /* Anchor list */\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Table of contents\n */\nexport type TableOfContents =\n | TableOfContentsBelowTablet\n | TableOfContentsAboveTablet\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount options\n */\ninterface MountOptions {\n header$: Observable<Header> /* Header observable */\n main$: Observable<Main> /* Main area observable */\n viewport$: Observable<Viewport> /* Viewport observable */\n tablet$: Observable<boolean> /* Tablet media observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount table of contents from source observable\n *\n * @param options - Options\n *\n * @return Operator function\n */\nexport function mountTableOfContents(\n { header$, main$, viewport$, tablet$ }: MountOptions\n): OperatorFunction<HTMLElement, TableOfContents> {\n return pipe(\n switchMap(el => tablet$\n .pipe(\n switchMap(tablet => {\n\n /* [tablet +]: Mount table of contents in sidebar */\n if (tablet) {\n const els = getElements<HTMLAnchorElement>(\".md-nav__link\", el)\n\n /* Watch and apply sidebar */\n const sidebar$ = watchSidebar(el, { main$, viewport$ })\n .pipe(\n applySidebar(el, { header$ })\n )\n\n /* Watch and apply anchor list (scroll spy) */\n const anchors$ = watchAnchorList(els, { header$, viewport$ })\n .pipe(\n applyAnchorList(els)\n )\n\n /* Combine into single hot observable */\n return combineLatest([sidebar$, anchors$])\n .pipe(\n map(([sidebar, anchors]) => ({ sidebar, anchors }))\n )\n\n /* [tablet -]: Unmount table of contents */\n } else {\n return of({})\n }\n })\n )\n )\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { reverse } from \"ramda\"\r\nimport {\r\n MonoTypeOperatorFunction,\r\n Observable,\r\n animationFrameScheduler,\r\n combineLatest,\r\n pipe\r\n} from \"rxjs\"\r\nimport {\r\n bufferCount,\r\n distinctUntilChanged,\r\n distinctUntilKeyChanged,\r\n finalize,\r\n map,\r\n observeOn,\r\n scan,\r\n startWith,\r\n switchMap,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport { Viewport, getElement, watchElementSize } from \"browser\"\r\n\r\nimport { Header } from \"../../../header\"\r\nimport { AnchorList } from \"../_\"\r\nimport {\r\n resetAnchorActive,\r\n resetAnchorBlur,\r\n setAnchorActive,\r\n setAnchorBlur\r\n} from \"../set\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch options\r\n */\r\ninterface WatchOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch anchor list\r\n *\r\n * This is effectively a scroll-spy implementation which will account for the\r\n * fixed header and automatically re-calculate anchor offsets when the viewport\r\n * is resized. The returned observable will only emit if the anchor list needs\r\n * to be repainted.\r\n *\r\n * This implementation tracks an anchor element's entire path starting from its\r\n * level up to the top-most anchor element, e.g. `[h3, h2, h1]`. Although the\r\n * Material theme currently doesn't make use of this information, it enables\r\n * the styling of the entire hierarchy through customization.\r\n *\r\n * Note that the current anchor is the last item of the `prev` anchor list.\r\n *\r\n * @param els - Anchor elements\r\n * @param options - Options\r\n *\r\n * @return Anchor list observable\r\n */\r\nexport function watchAnchorList(\r\n els: HTMLAnchorElement[], { header$, viewport$ }: WatchOptions\r\n): Observable<AnchorList> {\r\n const table = new Map<HTMLAnchorElement, HTMLElement>()\r\n for (const el of els) {\r\n const id = decodeURIComponent(el.hash.substring(1))\r\n const target = getElement(`[id=\"${id}\"]`)\r\n if (typeof target !== \"undefined\")\r\n table.set(el, target)\r\n }\r\n\r\n /* Compute necessary adjustment for header */\r\n const adjust$ = header$\r\n .pipe(\r\n map(header => 18 + header.height)\r\n )\r\n\r\n /* Compute partition of previous and next anchors */\r\n const partition$ = watchElementSize(document.body)\r\n .pipe(\r\n distinctUntilKeyChanged(\"height\"),\r\n\r\n /* Build index to map anchor paths to vertical offsets */\r\n map(() => {\r\n let path: HTMLAnchorElement[] = []\r\n return [...table].reduce((index, [anchor, target]) => {\r\n while (path.length) {\r\n const last = table.get(path[path.length - 1])!\r\n if (last.tagName >= target.tagName) {\r\n path.pop()\r\n } else {\r\n break\r\n }\r\n }\r\n\r\n /* If the current anchor is hidden, continue with its parent */\r\n let offset = target.offsetTop\r\n while (!offset && target.parentElement) {\r\n target = target.parentElement\r\n offset = target.offsetTop\r\n }\r\n\r\n /* Map reversed anchor path to vertical offset */\r\n return index.set(\r\n reverse(path = [...path, anchor]),\r\n offset\r\n )\r\n }, new Map<HTMLAnchorElement[], number>())\r\n }),\r\n\r\n /* Re-compute partition when viewport offset changes */\r\n switchMap(index => combineLatest([adjust$, viewport$])\r\n .pipe(\r\n scan(([prev, next], [adjust, { offset: { y } }]) => {\r\n\r\n /* Look forward */\r\n while (next.length) {\r\n const [, offset] = next[0]\r\n if (offset - adjust < y) {\r\n prev = [...prev, next.shift()!]\r\n } else {\r\n break\r\n }\r\n }\r\n\r\n /* Look backward */\r\n while (prev.length) {\r\n const [, offset] = prev[prev.length - 1]\r\n if (offset - adjust >= y) {\r\n next = [prev.pop()!, ...next]\r\n } else {\r\n break\r\n }\r\n }\r\n\r\n /* Return partition */\r\n return [prev, next]\r\n }, [[], [...index]]),\r\n distinctUntilChanged((a, b) => {\r\n return a[0] === b[0]\r\n && a[1] === b[1]\r\n })\r\n )\r\n )\r\n )\r\n\r\n /* Compute and return anchor list migrations */\r\n return partition$\r\n .pipe(\r\n map(([prev, next]) => ({\r\n prev: prev.map(([path]) => path),\r\n next: next.map(([path]) => path)\r\n })),\r\n\r\n /* Extract anchor list migrations */\r\n startWith({ prev: [], next: [] }),\r\n bufferCount(2, 1),\r\n map(([a, b]) => {\r\n\r\n /* Moving down */\r\n if (a.prev.length < b.prev.length) {\r\n return {\r\n prev: b.prev.slice(Math.max(0, a.prev.length - 1), b.prev.length),\r\n next: []\r\n }\r\n\r\n /* Moving up */\r\n } else {\r\n return {\r\n prev: b.prev.slice(-1),\r\n next: b.next.slice(0, b.next.length - a.next.length)\r\n }\r\n }\r\n })\r\n )\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Apply anchor list\r\n *\r\n * @param els - Anchor elements\r\n *\r\n * @return Operator function\r\n */\r\nexport function applyAnchorList(\r\n els: HTMLAnchorElement[]\r\n): MonoTypeOperatorFunction<AnchorList> {\r\n return pipe(\r\n\r\n /* Defer repaint to next animation frame */\r\n observeOn(animationFrameScheduler),\r\n tap(({ prev, next }) => {\r\n\r\n /* Look forward */\r\n for (const [el] of next) {\r\n resetAnchorActive(el)\r\n resetAnchorBlur(el)\r\n }\r\n\r\n /* Look backward */\r\n prev.forEach(([el], index) => {\r\n setAnchorActive(el, index === prev.length - 1)\r\n setAnchorBlur(el, true)\r\n })\r\n }),\r\n\r\n /* Reset on complete or error */\r\n finalize(() => {\r\n for (const el of els) {\r\n resetAnchorActive(el)\r\n resetAnchorBlur(el)\r\n }\r\n })\r\n )\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { Observable, OperatorFunction, combineLatest, pipe } from \"rxjs\"\nimport {\n filter,\n map,\n mapTo,\n sample,\n startWith,\n switchMap,\n take\n} from \"rxjs/operators\"\n\nimport { WorkerHandler } from \"browser\"\nimport {\n SearchMessage,\n SearchResult,\n isSearchQueryMessage,\n isSearchReadyMessage\n} from \"integrations/search\"\n\nimport { SearchQuery } from \"../query\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Search status\n */\nexport type SearchStatus =\n | \"waiting\" /* Search waiting for initialization */\n | \"ready\" /* Search ready */\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Search\n */\nexport interface Search {\n status: SearchStatus /* Search status */\n query: SearchQuery /* Search query */\n result: SearchResult[] /* Search result list */\n}\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount options\n */\ninterface MountOptions {\n query$: Observable<SearchQuery> /* Search query observable */\n reset$: Observable<void> /* Search reset observable */\n result$: Observable<SearchResult[]> /* Search result observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount search from source observable\n *\n * @param handler - Worker handler\n * @param options - Options\n *\n * @return Operator function\n */\nexport function mountSearch(\n { rx$, tx$ }: WorkerHandler<SearchMessage>,\n { query$, reset$, result$ }: MountOptions\n): OperatorFunction<HTMLElement, Search> {\n return pipe(\n switchMap(() => {\n\n /* Compute search status */\n const status$ = rx$\n .pipe(\n filter(isSearchReadyMessage),\n mapTo<SearchStatus>(\"ready\"),\n startWith(\"waiting\")\n ) as Observable<SearchStatus>\n\n /* Re-emit the latest query when search is ready */\n tx$\n .pipe(\n filter(isSearchQueryMessage),\n sample(status$),\n take(1)\n )\n .subscribe(tx$.next.bind(tx$))\n\n /* Combine into single observable */\n return combineLatest([status$, query$, result$, reset$])\n .pipe(\n map(([status, query, result]) => ({\n status,\n query,\n result\n }))\n )\n })\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { OperatorFunction, pipe } from \"rxjs\"\r\nimport {\r\n distinctUntilKeyChanged,\r\n map,\r\n switchMap\r\n} from \"rxjs/operators\"\r\n\r\nimport { WorkerHandler, setToggle } from \"browser\"\r\nimport {\r\n SearchMessage,\r\n SearchMessageType,\r\n SearchQueryMessage,\r\n SearchTransformFn\r\n} from \"integrations\"\r\n\r\nimport { watchSearchQuery } from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Search query\r\n */\r\nexport interface SearchQuery {\r\n value: string /* Query value */\r\n focus: boolean /* Query focus */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n transform?: SearchTransformFn /* Transformation function */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount search query from source observable\r\n *\r\n * @param handler - Worker handler\r\n * @param options - Options\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountSearchQuery(\r\n { tx$ }: WorkerHandler<SearchMessage>, options: MountOptions = {}\r\n): OperatorFunction<HTMLInputElement, SearchQuery> {\r\n return pipe(\r\n switchMap(el => {\r\n const query$ = watchSearchQuery(el, options)\r\n\r\n /* Subscribe worker to search query */\r\n query$\r\n .pipe(\r\n distinctUntilKeyChanged(\"value\"),\r\n map(({ value }): SearchQueryMessage => ({\r\n type: SearchMessageType.QUERY,\r\n data: value\r\n }))\r\n )\r\n .subscribe(tx$.next.bind(tx$))\r\n\r\n /* Toggle search on focus */\r\n query$\r\n .pipe(\r\n distinctUntilKeyChanged(\"focus\")\r\n )\r\n .subscribe(({ focus }) => {\r\n if (focus)\r\n setToggle(\"search\", focus)\r\n })\r\n\r\n /* Return search query */\r\n return query$\r\n })\r\n )\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { Observable, combineLatest, fromEvent, merge } from \"rxjs\"\nimport {\n delay,\n distinctUntilChanged,\n map,\n startWith\n} from \"rxjs/operators\"\n\nimport { watchElementFocus } from \"browser\"\nimport { SearchTransformFn, defaultTransform } from \"integrations\"\n\nimport { SearchQuery } from \"../_\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch options\n */\ninterface WatchOptions {\n transform?: SearchTransformFn /* Transformation function */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Watch search query\n *\n * Note that the focus event which triggers re-reading the current query value\n * is delayed by `1ms` so the input's empty state is allowed to propagate.\n *\n * @param el - Search query element\n * @param options - Options\n *\n * @return Search query observable\n */\nexport function watchSearchQuery(\n el: HTMLInputElement, { transform }: WatchOptions = {}\n): Observable<SearchQuery> {\n const fn = transform || defaultTransform\n\n /* Intercept keyboard events */\n const value$ = merge(\n fromEvent(el, \"keyup\"),\n fromEvent(el, \"focus\").pipe(delay(1))\n )\n .pipe(\n map(() => fn(el.value)),\n startWith(fn(el.value)),\n distinctUntilChanged()\n )\n\n /* Intercept focus events */\n const focus$ = watchElementFocus(el)\n\n /* Combine into single observable */\n return combineLatest([value$, focus$])\n .pipe(\n map(([value, focus]) => ({ value, focus }))\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { OperatorFunction, pipe } from \"rxjs\"\r\nimport {\r\n mapTo,\r\n startWith,\r\n switchMap,\r\n switchMapTo,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport { setElementFocus } from \"browser\"\r\n\r\nimport { useComponent } from \"../../../_\"\r\nimport { watchSearchReset } from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount search reset from source observable\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountSearchReset(): OperatorFunction<HTMLElement, void> {\r\n return pipe(\r\n switchMap(el => watchSearchReset(el)\r\n .pipe(\r\n switchMapTo(useComponent(\"search-query\")),\r\n tap(setElementFocus),\r\n mapTo(undefined)\r\n )\r\n ),\r\n startWith(undefined)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, fromEvent } from \"rxjs\"\r\nimport { mapTo } from \"rxjs/operators\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch search reset\r\n *\r\n * @param el - Search reset element\r\n *\r\n * @return Search reset observable\r\n */\r\nexport function watchSearchReset(\r\n el: HTMLElement\r\n): Observable<void> {\r\n return fromEvent(el, \"click\")\r\n .pipe(\r\n mapTo(undefined)\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { translate } from \"utilities\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set number of search results\r\n *\r\n * @param el - Search result metadata element\r\n * @param value - Number of results\r\n */\r\nexport function setSearchResultMeta(\r\n el: HTMLElement, value: number\r\n): void {\r\n switch (value) {\r\n\r\n /* No results */\r\n case 0:\r\n el.textContent = translate(\"search.result.none\")\r\n break\r\n\r\n /* One result */\r\n case 1:\r\n el.textContent = translate(\"search.result.one\")\r\n break\r\n\r\n /* Multiple result */\r\n default:\r\n el.textContent = translate(\"search.result.other\", value.toString())\r\n }\r\n}\r\n\r\n/**\r\n * Reset number of search results\r\n *\r\n * @param el - Search result metadata element\r\n */\r\nexport function resetSearchResultMeta(\r\n el: HTMLElement\r\n): void {\r\n el.textContent = translate(\"search.result.placeholder\")\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Add an element to the search result list\r\n *\r\n * @param el - Search result list element\r\n * @param child - Search result element\r\n */\r\nexport function addToSearchResultList(\r\n el: HTMLElement, child: Element\r\n): void {\r\n el.appendChild(child)\r\n}\r\n\r\n/**\r\n * Reset search result list\r\n *\r\n * @param el - Search result list element\r\n */\r\nexport function resetSearchResultList(\r\n el: HTMLElement\r\n): void {\r\n el.innerHTML = \"\"\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport {\n MonoTypeOperatorFunction,\n Observable,\n animationFrameScheduler,\n pipe\n} from \"rxjs\"\nimport {\n finalize,\n map,\n mapTo,\n observeOn,\n scan,\n switchMap,\n withLatestFrom\n} from \"rxjs/operators\"\n\nimport { getElementOrThrow } from \"browser\"\nimport { SearchResult } from \"integrations/search\"\nimport { renderSearchResult } from \"templates\"\n\nimport { SearchQuery } from \"../../query\"\nimport {\n addToSearchResultList,\n resetSearchResultList,\n resetSearchResultMeta,\n setSearchResultMeta\n} from \"../set\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Apply options\n */\ninterface ApplyOptions {\n query$: Observable<SearchQuery> /* Search query observable */\n ready$: Observable<boolean> /* Search ready observable */\n fetch$: Observable<boolean> /* Result fetch observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Apply search results\n *\n * This function will perform a lazy rendering of the search results, depending\n * on the vertical offset of the search result container. When the scroll offset\n * reaches the bottom of the element, more results are fetched and rendered.\n *\n * @param el - Search result element\n * @param options - Options\n *\n * @return Operator function\n */\nexport function applySearchResult(\n el: HTMLElement, { query$, ready$, fetch$ }: ApplyOptions\n): MonoTypeOperatorFunction<SearchResult[]> {\n const list = getElementOrThrow(\".md-search-result__list\", el)\n const meta = getElementOrThrow(\".md-search-result__meta\", el)\n return pipe(\n\n /* Apply search result metadata */\n withLatestFrom(query$, ready$),\n map(([result, query]) => {\n if (query.value) {\n setSearchResultMeta(meta, result.length)\n } else {\n resetSearchResultMeta(meta)\n }\n return result\n }),\n\n /* Apply search result list */\n switchMap(result => fetch$\n .pipe(\n\n /* Defer repaint to next animation frame */\n observeOn(animationFrameScheduler),\n scan(index => {\n const container = el.parentElement!\n while (index < result.length) {\n addToSearchResultList(list, renderSearchResult(result[index++]))\n if (container.scrollHeight - container.offsetHeight > 16)\n break\n }\n return index\n }, 0),\n\n /* Re-map to search result */\n mapTo(result),\n\n /* Reset on complete or error */\n finalize(() => {\n resetSearchResultList(list)\n })\n )\n )\n )\n}\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { identity } from \"ramda\"\nimport { Observable, OperatorFunction, pipe } from \"rxjs\"\nimport {\n distinctUntilChanged,\n filter,\n map,\n mapTo,\n pluck,\n startWith,\n switchMap\n} from \"rxjs/operators\"\n\nimport { WorkerHandler, watchElementOffset } from \"browser\"\nimport {\n SearchMessage,\n SearchResult,\n isSearchReadyMessage,\n isSearchResultMessage\n} from \"integrations\"\n\nimport { SearchQuery } from \"../../query\"\nimport { applySearchResult } from \"../react\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount options\n */\ninterface MountOptions {\n query$: Observable<SearchQuery> /* Search query observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount search result from source observable\n *\n * @param handler - Worker handler\n * @param options - Options\n *\n * @return Operator function\n */\nexport function mountSearchResult(\n { rx$ }: WorkerHandler<SearchMessage>, { query$ }: MountOptions\n): OperatorFunction<HTMLElement, SearchResult[]> {\n return pipe(\n switchMap(el => {\n const container = el.parentElement!\n\n /* Compute if search is ready */\n const ready$ = rx$\n .pipe(\n filter(isSearchReadyMessage),\n mapTo(true)\n )\n\n /* Compute whether there are more search results to fetch */\n const fetch$ = watchElementOffset(container)\n .pipe(\n map(({ y }) => {\n return y >= container.scrollHeight - container.offsetHeight - 16\n }),\n distinctUntilChanged(),\n filter(identity)\n )\n\n /* Apply search results */\n return rx$\n .pipe(\n filter(isSearchResultMessage),\n pluck(\"data\"),\n applySearchResult(el, { query$, ready$, fetch$ }),\n startWith([])\n )\n })\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, OperatorFunction, Subject, pipe } from \"rxjs\"\r\nimport { distinctUntilKeyChanged, switchMap, tap } from \"rxjs/operators\"\r\n\r\nimport { Viewport } from \"browser\"\r\n\r\nimport { useComponent } from \"../../_\"\r\nimport { Header } from \"../../header\"\r\nimport {\r\n applyHeaderShadow,\r\n watchMain\r\n} from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Main area\r\n */\r\nexport interface Main {\r\n offset: number /* Main area top offset */\r\n height: number /* Main area visible height */\r\n active: boolean /* Scrolled past top offset */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount main area from source observable\r\n *\r\n * The header must be connected to the main area observable outside of the\r\n * operator function, as the header will persist in-between document switches\r\n * while the main area is replaced. However, the header observable must be\r\n * passed to this function, so we connect both via a long-living subject.\r\n *\r\n * @param options - Options\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountMain(\r\n { header$, viewport$ }: MountOptions\r\n): OperatorFunction<HTMLElement, Main> {\r\n const main$ = new Subject<Main>()\r\n\r\n /* Connect to main area observable via long-living subject */\r\n useComponent(\"header\")\r\n .pipe(\r\n switchMap(header => main$\r\n .pipe(\r\n distinctUntilKeyChanged(\"active\"),\r\n applyHeaderShadow(header)\r\n )\r\n )\r\n )\r\n .subscribe()\r\n\r\n /* Return operator */\r\n return pipe(\r\n switchMap(el => watchMain(el, { header$, viewport$ })),\r\n tap(main => main$.next(main))\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport {\r\n MonoTypeOperatorFunction,\r\n Observable,\r\n animationFrameScheduler,\r\n combineLatest,\r\n pipe\r\n} from \"rxjs\"\r\nimport {\r\n distinctUntilChanged,\r\n distinctUntilKeyChanged,\r\n finalize,\r\n map,\r\n observeOn,\r\n pluck,\r\n shareReplay,\r\n switchMap,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport { Viewport, watchElementSize } from \"browser\"\r\n\r\nimport { Header } from \"../../header\"\r\nimport { Main } from \"../_\"\r\nimport {\r\n resetHeaderShadow,\r\n setHeaderShadow\r\n} from \"../set\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch options\r\n */\r\ninterface WatchOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch main area\r\n *\r\n * This function returns an observable that computes the visual parameters of\r\n * the main area which depends on the viewport vertical offset and height, as\r\n * well as the height of the header element, if the header is fixed.\r\n *\r\n * @param el - Main area element\r\n * @param options - Options\r\n *\r\n * @return Main area observable\r\n */\r\nexport function watchMain(\r\n el: HTMLElement, { header$, viewport$ }: WatchOptions\r\n): Observable<Main> {\r\n\r\n /* Compute necessary adjustment for header */\r\n const adjust$ = header$\r\n .pipe(\r\n pluck(\"height\"),\r\n distinctUntilChanged(),\r\n shareReplay(1)\r\n )\r\n\r\n /* Compute the main area's top and bottom borders */\r\n const border$ = adjust$\r\n .pipe(\r\n switchMap(() => watchElementSize(el)\r\n .pipe(\r\n map(({ height }) => ({\r\n top: el.offsetTop,\r\n bottom: el.offsetTop + height\r\n }))\r\n )\r\n ),\r\n distinctUntilKeyChanged(\"bottom\"),\r\n shareReplay(1)\r\n )\r\n\r\n /* Compute the main area's offset, visible height and if we scrolled past */\r\n return combineLatest([adjust$, border$, viewport$])\r\n .pipe(\r\n map(([header, { top, bottom }, { offset: { y }, size: { height } }]) => {\r\n height = Math.max(0, height\r\n - Math.max(0, top - y, header)\r\n - Math.max(0, height + y - bottom)\r\n )\r\n return {\r\n offset: top - header,\r\n height,\r\n active: top - header <= y\r\n }\r\n }),\r\n distinctUntilChanged<Main>((a, b) => {\r\n return a.offset === b.offset\r\n && a.height === b.height\r\n && a.active === b.active\r\n })\r\n )\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Apply header shadow\r\n *\r\n * @param el - Header element\r\n *\r\n * @return Operator function\r\n */\r\nexport function applyHeaderShadow(\r\n el: HTMLElement\r\n): MonoTypeOperatorFunction<Main> {\r\n return pipe(\r\n\r\n /* Defer repaint to next animation frame */\r\n observeOn(animationFrameScheduler),\r\n tap(({ active }) => {\r\n setHeaderShadow(el, active)\r\n }),\r\n\r\n /* Reset on complete or error */\r\n finalize(() => {\r\n resetHeaderShadow(el)\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set header shadow\r\n *\r\n * @param el - Header element\r\n * @param value - Whether the shadow is shown\r\n */\r\nexport function setHeaderShadow(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.setAttribute(\"data-md-state\", value ? \"shadow\" : \"\")\r\n}\r\n\r\n/**\r\n * Reset header shadow\r\n *\r\n * @param el - Header element\r\n */\r\nexport function resetHeaderShadow(\r\n el: HTMLElement\r\n): void {\r\n el.removeAttribute(\"data-md-state\")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, OperatorFunction, pipe } from \"rxjs\"\r\nimport {\r\n distinctUntilKeyChanged,\r\n map,\r\n switchMap\r\n} from \"rxjs/operators\"\r\n\r\nimport { Viewport, watchViewportAt } from \"browser\"\r\n\r\nimport { Header } from \"../../header\"\r\nimport { applyHero } from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Hero\r\n */\r\nexport interface Hero {\r\n hidden: boolean /* Whether the hero is hidden */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount hero from source observable\r\n *\r\n * @param options - Options\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountHero(\r\n { header$, viewport$ }: MountOptions\r\n): OperatorFunction<HTMLElement, Hero> {\r\n return pipe(\r\n switchMap(el => watchViewportAt(el, { header$, viewport$ })\r\n .pipe(\r\n map(({ offset: { y } }) => ({ hidden: y >= 20 })),\r\n distinctUntilKeyChanged(\"hidden\"),\r\n applyHero(el)\r\n )\r\n )\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport {\r\n MonoTypeOperatorFunction,\r\n animationFrameScheduler,\r\n pipe\r\n} from \"rxjs\"\r\nimport { finalize, observeOn, tap } from \"rxjs/operators\"\r\n\r\nimport { Hero } from \"../_\"\r\nimport {\r\n resetHeroHidden,\r\n setHeroHidden\r\n} from \"../set\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Apply hero\r\n *\r\n * @param el - Hero element\r\n *\r\n * @return Operator function\r\n */\r\nexport function applyHero(\r\n el: HTMLElement\r\n): MonoTypeOperatorFunction<Hero> {\r\n return pipe(\r\n\r\n /* Defer repaint to next animation frame */\r\n observeOn(animationFrameScheduler),\r\n tap(({ hidden }) => {\r\n setHeroHidden(el, hidden)\r\n }),\r\n\r\n /* Reset on complete or error */\r\n finalize(() => {\r\n resetHeroHidden(el)\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set hero hidden\r\n *\r\n * @param el - Hero element\r\n * @param value - Whether the element is hidden\r\n */\r\nexport function setHeroHidden(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.setAttribute(\"data-md-state\", value ? \"hidden\" : \"\")\r\n}\r\n\r\n/**\r\n * Reset hero hidden\r\n *\r\n * @param el - Hero element\r\n */\r\nexport function resetHeroHidden(\r\n el: HTMLElement\r\n): void {\r\n el.removeAttribute(\"data-md-state\")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, OperatorFunction, combineLatest, pipe } from \"rxjs\"\r\nimport {\r\n distinctUntilChanged,\r\n filter,\r\n map,\r\n shareReplay,\r\n startWith,\r\n switchMap,\r\n withLatestFrom\r\n} from \"rxjs/operators\"\r\n\r\nimport {\r\n Viewport,\r\n getElement,\r\n watchViewportAt\r\n} from \"browser\"\r\n\r\nimport { useComponent } from \"../../_\"\r\nimport {\r\n applyHeaderType,\r\n watchHeader\r\n} from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Header type\r\n */\r\nexport type HeaderType =\r\n | \"site\" /* Header shows site title */\r\n | \"page\" /* Header shows page title */\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Header\r\n */\r\nexport interface Header {\r\n type: HeaderType /* Header type */\r\n sticky: boolean /* Header stickyness */\r\n height: number /* Header visible height */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n document$: Observable<Document> /* Document observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount header from source observable\r\n *\r\n * @param options - Options\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountHeader(\r\n { document$, viewport$ }: MountOptions\r\n): OperatorFunction<HTMLElement, Header> {\r\n return pipe(\r\n switchMap(el => {\r\n const header$ = watchHeader(el, { document$ })\r\n\r\n /* Compute whether the header should switch to page header */\r\n const type$ = useComponent(\"main\")\r\n .pipe(\r\n map(main => getElement(\"h1, h2, h3, h4, h5, h6\", main)!),\r\n filter(hx => typeof hx !== \"undefined\"),\r\n withLatestFrom(useComponent(\"header-title\")),\r\n switchMap(([hx, title]) => watchViewportAt(hx, { header$, viewport$ })\r\n .pipe(\r\n map(({ offset: { y } }) => {\r\n return y >= hx.offsetHeight ? \"page\" : \"site\"\r\n }),\r\n distinctUntilChanged(),\r\n applyHeaderType(title)\r\n )\r\n ),\r\n startWith<HeaderType>(\"site\")\r\n )\r\n\r\n /* Combine into single observable */\r\n return combineLatest([header$, type$])\r\n .pipe(\r\n map(([header, type]): Header => ({ type, ...header })),\r\n shareReplay(1)\r\n )\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport {\r\n MonoTypeOperatorFunction,\r\n Observable,\r\n animationFrameScheduler,\r\n of,\r\n pipe\r\n} from \"rxjs\"\r\nimport {\r\n distinctUntilChanged,\r\n finalize,\r\n map,\r\n observeOn,\r\n shareReplay,\r\n switchMap,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport { watchElementSize } from \"browser\"\r\n\r\nimport { Header, HeaderType } from \"../_\"\r\nimport {\r\n resetHeaderTitleActive,\r\n setHeaderTitleActive\r\n} from \"../set\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch options\r\n */\r\ninterface WatchOptions {\r\n document$: Observable<Document> /* Document observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Watch header\r\n *\r\n * @param el - Header element\r\n *\r\n * @return Header observable\r\n */\r\nexport function watchHeader(\r\n el: HTMLElement, { document$ }: WatchOptions\r\n): Observable<Omit<Header, \"type\">> {\r\n return document$\r\n .pipe(\r\n map(() => {\r\n const styles = getComputedStyle(el)\r\n return [\r\n \"sticky\", /* Modern browsers */\r\n \"-webkit-sticky\" /* Safari */\r\n ].includes(styles.position)\r\n }),\r\n distinctUntilChanged(),\r\n switchMap(sticky => {\r\n if (sticky) {\r\n return watchElementSize(el)\r\n .pipe(\r\n map(({ height }) => ({\r\n sticky: true,\r\n height\r\n }))\r\n )\r\n } else {\r\n return of({\r\n sticky: false,\r\n height: 0\r\n })\r\n }\r\n }),\r\n shareReplay(1)\r\n )\r\n}\r\n\r\n/* ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Apply header title type\r\n *\r\n * @param el - Header title element\r\n *\r\n * @return Operator function\r\n */\r\nexport function applyHeaderType(\r\n el: HTMLElement\r\n): MonoTypeOperatorFunction<HeaderType> {\r\n return pipe(\r\n\r\n /* Defer repaint to next animation frame */\r\n observeOn(animationFrameScheduler),\r\n tap(type => {\r\n setHeaderTitleActive(el, type === \"page\")\r\n }),\r\n\r\n /* Reset on complete or error */\r\n finalize(() => {\r\n resetHeaderTitleActive(el)\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set header title active\r\n *\r\n * @param el - Header title element\r\n * @param value - Whether the title is shown\r\n */\r\nexport function setHeaderTitleActive(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.setAttribute(\"data-md-state\", value ? \"active\" : \"\")\r\n}\r\n\r\n/**\r\n * Reset header title active\r\n *\r\n * @param el - Header title element\r\n */\r\nexport function resetHeaderTitleActive(\r\n el: HTMLElement\r\n): void {\r\n el.removeAttribute(\"data-md-state\")\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable, OperatorFunction, of, pipe } from \"rxjs\"\r\nimport {\r\n distinctUntilKeyChanged,\r\n map,\r\n switchMap\r\n} from \"rxjs/operators\"\r\n\r\nimport { Viewport, watchViewportAt } from \"browser\"\r\n\r\nimport { Header } from \"../../header\"\r\nimport { applyTabs } from \"../react\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Tabs\r\n */\r\nexport interface Tabs {\r\n hidden: boolean /* Whether the tabs are hidden */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n header$: Observable<Header> /* Header observable */\r\n viewport$: Observable<Viewport> /* Viewport observable */\r\n screen$: Observable<boolean> /* Media screen observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount tabs from source observable\r\n *\r\n * @param options - Options\r\n *\r\n * @return Operator function\r\n */\r\nexport function mountTabs(\r\n { header$, viewport$, screen$ }: MountOptions\r\n): OperatorFunction<HTMLElement, Tabs> {\r\n return pipe(\r\n switchMap(el => screen$\r\n .pipe(\r\n switchMap(screen => {\r\n\r\n /* [screen +]: Mount tabs above screen breakpoint */\r\n if (screen) {\r\n return watchViewportAt(el, { header$, viewport$ })\r\n .pipe(\r\n map(({ offset: { y } }) => ({ hidden: y >= 10 })),\r\n distinctUntilKeyChanged(\"hidden\"),\r\n applyTabs(el)\r\n )\r\n\r\n /* [screen -]: Unmount tabs below screen breakpoint */\r\n } else {\r\n return of({ hidden: true })\r\n }\r\n })\r\n )\r\n )\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport {\r\n MonoTypeOperatorFunction,\r\n animationFrameScheduler,\r\n pipe\r\n} from \"rxjs\"\r\nimport { finalize, observeOn, tap } from \"rxjs/operators\"\r\n\r\nimport { Tabs } from \"../_\"\r\nimport {\r\n resetTabsHidden,\r\n setTabsHidden\r\n} from \"../set\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Apply tabs\r\n *\r\n * @param el - Tabs element\r\n *\r\n * @return Operator function\r\n */\r\nexport function applyTabs(\r\n el: HTMLElement\r\n): MonoTypeOperatorFunction<Tabs> {\r\n return pipe(\r\n\r\n /* Defer repaint to next animation frame */\r\n observeOn(animationFrameScheduler),\r\n tap(({ hidden }) => {\r\n setTabsHidden(el, hidden)\r\n }),\r\n\r\n /* Reset on complete or error */\r\n finalize(() => {\r\n resetTabsHidden(el)\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Set tabs hidden\r\n *\r\n * @param el - Tabs element\r\n * @param value - Whether the element is hidden\r\n */\r\nexport function setTabsHidden(\r\n el: HTMLElement, value: boolean\r\n): void {\r\n el.setAttribute(\"data-md-state\", value ? \"hidden\" : \"\")\r\n}\r\n\r\n/**\r\n * Reset tabs hidden\r\n *\r\n * @param el - Tabs element\r\n */\r\nexport function resetTabsHidden(\r\n el: HTMLElement\r\n): void {\r\n el.removeAttribute(\"data-md-state\")\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { Observable, OperatorFunction, of, pipe } from \"rxjs\"\nimport { map, switchMap } from \"rxjs/operators\"\n\nimport { Viewport } from \"browser\"\n\nimport { Header } from \"../../header\"\nimport { Main } from \"../../main\"\nimport {\n Sidebar,\n applySidebar,\n watchSidebar\n} from \"../../shared\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Navigation for [screen -]\n */\ninterface NavigationBelowScreen {} // tslint:disable-line\n\n/**\n * Navigation for [screen +]\n */\ninterface NavigationAboveScreen {\n sidebar: Sidebar /* Sidebar */\n}\n\n/* ------------------------------------------------------------------------- */\n\n/**\n * Navigation\n */\nexport type Navigation =\n | NavigationBelowScreen\n | NavigationAboveScreen\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount options\n */\ninterface MountOptions {\n header$: Observable<Header> /* Header observable */\n main$: Observable<Main> /* Main area observable */\n viewport$: Observable<Viewport> /* Viewport observable */\n screen$: Observable<boolean> /* Screen media observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount navigation from source observable\n *\n * @param options - Options\n *\n * @return Operator function\n */\nexport function mountNavigation(\n { header$, main$, viewport$, screen$ }: MountOptions\n): OperatorFunction<HTMLElement, Navigation> {\n return pipe(\n switchMap(el => screen$\n .pipe(\n switchMap(screen => {\n\n /* [screen +]: Mount navigation in sidebar */\n if (screen) {\n return watchSidebar(el, { main$, viewport$ })\n .pipe(\n applySidebar(el, { header$ }),\n map(sidebar => ({ sidebar }))\n )\n\n /* [screen -]: Mount navigation in drawer */\n } else {\n return of({})\n }\n })\n )\n )\n )\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { NEVER, Observable, fromEvent, iif, merge } from \"rxjs\"\r\nimport { map, mapTo, shareReplay, switchMap } from \"rxjs/operators\"\r\n\r\nimport { getElements } from \"browser\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch options\r\n */\r\ninterface PatchOptions {\r\n document$: Observable<Document> /* Document observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Check whether the given device is an Apple device\r\n *\r\n * @return Test result\r\n */\r\nfunction isAppleDevice(): boolean {\r\n return /(iPad|iPhone|iPod)/.test(navigator.userAgent)\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch all elements with `data-md-scrollfix` attributes\r\n *\r\n * This is a year-old patch which ensures that overflow scrolling works at the\r\n * top and bottom of containers on iOS by ensuring a `1px` scroll offset upon\r\n * the start of a touch event.\r\n *\r\n * @see https://bit.ly/2SCtAOO - Original source\r\n *\r\n * @param options - Options\r\n */\r\nexport function patchScrollfix(\r\n { document$ }: PatchOptions\r\n): void {\r\n const els$ = document$\r\n .pipe(\r\n map(() => getElements(\"[data-md-scrollfix]\")),\r\n shareReplay(1)\r\n )\r\n\r\n /* Remove marker attribute, so we'll only add the fix once */\r\n els$.subscribe(els => {\r\n for (const el of els)\r\n el.removeAttribute(\"data-md-scrollfix\")\r\n })\r\n\r\n /* Patch overflow scrolling on touch start */\r\n iif(isAppleDevice, els$, NEVER)\r\n .pipe(\r\n switchMap(els => merge(...els.map(el => (\r\n fromEvent(el, \"touchstart\", { passive: true })\r\n .pipe(\r\n mapTo(el)\r\n )\r\n ))))\r\n )\r\n .subscribe(el => {\r\n const top = el.scrollTop\r\n\r\n /* We're at the top of the container */\r\n if (top === 0) {\r\n el.scrollTop = 1\r\n\r\n /* We're at the bottom of the container */\r\n } else if (top + el.offsetHeight === el.scrollHeight) {\r\n el.scrollTop = top - 1\r\n }\r\n })\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { NEVER, Observable } from \"rxjs\"\nimport { catchError, map, switchMap } from \"rxjs/operators\"\n\nimport { getElementOrThrow, getElements } from \"browser\"\nimport { renderSource } from \"templates\"\nimport { cache, hash } from \"utilities\"\n\nimport { fetchSourceFactsFromGitHub } from \"./github\"\nimport { fetchSourceFactsFromGitLab } from \"./gitlab\"\n\n/* ----------------------------------------------------------------------------\n * Types\n * ------------------------------------------------------------------------- */\n\n/**\n * Source facts\n */\nexport type SourceFacts = string[]\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Patch options\n */\ninterface PatchOptions {\n document$: Observable<Document> /* Document observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Helper functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Fetch source facts\n *\n * @param url - Source repository URL\n *\n * @return Source facts observable\n */\nfunction fetchSourceFacts(\n url: string\n): Observable<SourceFacts> {\n const [type] = url.match(/(git(?:hub|lab))/i) || []\n switch (type.toLowerCase()) {\n\n /* GitHub repository */\n case \"github\":\n const [, user, repo] = url.match(/^.+github\\.com\\/([^\\/]+)\\/?([^\\/]+)/i)\n return fetchSourceFactsFromGitHub(user, repo)\n\n /* GitLab repository */\n case \"gitlab\":\n const [, base, slug] = url.match(/^.+?([^\\/]*gitlab[^\\/]+)\\/(.+?)\\/?$/i)\n return fetchSourceFactsFromGitLab(base, slug)\n\n /* Everything else */\n default:\n return NEVER\n }\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Patch elements containing repository information\n *\n * This function will retrieve the URL from the repository link and try to\n * query data from integrated source code platforms like GitHub or GitLab.\n *\n * @param options - Options\n */\nexport function patchSource(\n { document$ }: PatchOptions\n): void {\n document$\n .pipe(\n map(() => getElementOrThrow<HTMLAnchorElement>(\".md-source[href]\")),\n switchMap(({ href }) => (\n cache(`${hash(href)}`, () => fetchSourceFacts(href))\n )),\n catchError(() => NEVER)\n )\n .subscribe(facts => {\n for (const el of getElements(\".md-source__repository\")) {\n if (!el.hasAttribute(\"data-md-state\")) {\n el.setAttribute(\"data-md-state\", \"done\")\n el.appendChild(renderSource(facts))\n }\n }\n })\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Repo, User } from \"github-types\"\r\nimport { Observable, of } from \"rxjs\"\r\nimport { ajax } from \"rxjs/ajax\"\r\nimport { filter, pluck, switchMap } from \"rxjs/operators\"\r\n\r\nimport { round } from \"utilities\"\r\n\r\nimport { SourceFacts } from \"..\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Fetch GitHub source facts\r\n *\r\n * @param user - GitHub user\r\n * @param repo - GitHub repository\r\n *\r\n * @return Source facts observable\r\n */\r\nexport function fetchSourceFactsFromGitHub(\r\n user: string, repo?: string\r\n): Observable<SourceFacts> {\r\n return ajax({\r\n url: typeof repo !== \"undefined\"\r\n ? `https://api.github.com/repos/${user}/${repo}`\r\n : `https://api.github.com/users/${user}`,\r\n responseType: \"json\"\r\n })\r\n .pipe(\r\n filter(({ status }) => status === 200),\r\n pluck(\"response\"),\r\n switchMap(data => {\r\n\r\n /* GitHub repository */\r\n if (typeof repo !== \"undefined\") {\r\n const { stargazers_count, forks_count }: Repo = data\r\n return of([\r\n `${round(stargazers_count || 0)} Stars`,\r\n `${round(forks_count || 0)} Forks`\r\n ])\r\n\r\n /* GitHub user/organization */\r\n } else {\r\n const { public_repos }: User = data\r\n return of([\r\n `${round(public_repos || 0)} Repositories`\r\n ])\r\n }\r\n })\r\n )\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { ProjectSchema } from \"gitlab\"\r\nimport { Observable } from \"rxjs\"\r\nimport { ajax } from \"rxjs/ajax\"\r\nimport { filter, map, pluck } from \"rxjs/operators\"\r\n\r\nimport { round } from \"utilities\"\r\n\r\nimport { SourceFacts } from \"..\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Fetch GitLab source facts\r\n *\r\n * @param base - GitLab base\r\n * @param project - GitLab project\r\n *\r\n * @return Source facts observable\r\n */\r\nexport function fetchSourceFactsFromGitLab(\r\n base: string, project: string\r\n): Observable<SourceFacts> {\r\n return ajax({\r\n url: `https://${base}/api/v4/projects/${encodeURIComponent(project)}`,\r\n responseType: \"json\"\r\n })\r\n .pipe(\r\n filter(({ status }) => status === 200),\r\n pluck(\"response\"),\r\n map(({ star_count, forks_count }: ProjectSchema) => ([\r\n `${round(star_count)} Stars`,\r\n `${round(forks_count)} Forks`\r\n ]))\r\n )\r\n}\r\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\n// DISCLAIMER: this file is still WIP. There're some refactoring opportunities\n// which must be tackled after we gathered some feedback on v5.\n// tslint:disable\n\nimport { sortBy, prop, values } from \"ramda\"\nimport {\n merge,\n combineLatest,\n animationFrameScheduler,\n fromEvent,\n from,\n defer,\n of,\n NEVER\n} from \"rxjs\"\nimport { ajax } from \"rxjs/ajax\"\nimport {\n delay,\n switchMap,\n tap,\n filter,\n withLatestFrom,\n observeOn,\n take,\n shareReplay,\n pluck,\n catchError,\n map\n} from \"rxjs/operators\"\n\nimport {\n watchToggle,\n setToggle,\n getElements,\n watchMedia,\n watchDocument,\n watchLocation,\n watchLocationHash,\n watchViewport,\n isLocalLocation,\n setLocationHash,\n watchLocationBase\n} from \"browser\"\nimport {\n mountHeader,\n mountHero,\n mountMain,\n mountNavigation,\n mountSearch,\n mountTableOfContents,\n mountTabs,\n useComponent,\n setupComponents,\n mountSearchQuery,\n mountSearchReset,\n mountSearchResult\n} from \"components\"\nimport {\n setupClipboard,\n setupDialog,\n setupKeyboard,\n setupInstantLoading,\n setupSearchWorker,\n SearchIndex\n} from \"integrations\"\nimport {\n patchCodeBlocks,\n patchTables,\n patchDetails,\n patchScrollfix,\n patchSource,\n patchScripts\n} from \"patches\"\nimport { isConfig } from \"utilities\"\n\n/* ------------------------------------------------------------------------- */\n\n/* Denote that JavaScript is available */\ndocument.documentElement.classList.remove(\"no-js\")\ndocument.documentElement.classList.add(\"js\")\n\n/* Test for iOS */\nif (navigator.userAgent.match(/(iPad|iPhone|iPod)/g))\n document.documentElement.classList.add(\"ios\")\n\n/**\n * Set scroll lock\n *\n * @param el - Scrollable element\n * @param value - Vertical offset\n */\nexport function setScrollLock(\n el: HTMLElement, value: number\n): void {\n el.setAttribute(\"data-md-state\", \"lock\")\n el.style.top = `-${value}px`\n}\n\n/**\n * Reset scroll lock\n *\n * @param el - Scrollable element\n */\nexport function resetScrollLock(\n el: HTMLElement\n): void {\n const value = -1 * parseInt(el.style.top, 10)\n el.removeAttribute(\"data-md-state\")\n el.style.top = \"\"\n if (value)\n window.scrollTo(0, value)\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Initialize Material for MkDocs\n *\n * @param config - Configuration\n */\nexport function initialize(config: unknown) {\n if (!isConfig(config))\n throw new SyntaxError(`Invalid configuration: ${JSON.stringify(config)}`)\n\n /* Set up subjects */\n const document$ = watchDocument()\n const location$ = watchLocation()\n\n /* Set up user interface observables */\n const base$ = watchLocationBase(config.base, { location$ })\n const hash$ = watchLocationHash()\n const viewport$ = watchViewport()\n const tablet$ = watchMedia(\"(min-width: 960px)\")\n const screen$ = watchMedia(\"(min-width: 1220px)\")\n\n /* ----------------------------------------------------------------------- */\n\n /* Set up component bindings */\n setupComponents([\n \"announce\", /* Announcement bar */\n \"container\", /* Container */\n \"header\", /* Header */\n \"header-title\", /* Header title */\n \"hero\", /* Hero */\n \"main\", /* Main area */\n \"navigation\", /* Navigation */\n \"search\", /* Search */\n \"search-query\", /* Search input */\n \"search-reset\", /* Search reset */\n \"search-result\", /* Search results */\n \"skip\", /* Skip link */\n \"tabs\", /* Tabs */\n \"toc\" /* Table of contents */\n ], { document$ })\n\n const keyboard$ = setupKeyboard()\n\n patchCodeBlocks({ document$, viewport$ })\n patchDetails({ document$, hash$ })\n patchScripts({ document$ })\n patchSource({ document$ })\n patchTables({ document$ })\n\n /* Force 1px scroll offset to trigger overflow scrolling */\n patchScrollfix({ document$ })\n\n /* Set up clipboard and dialog */\n const dialog$ = setupDialog()\n const clipboard$ = setupClipboard({ document$, dialog$ })\n\n /* ----------------------------------------------------------------------- */\n\n /* Create header observable */\n const header$ = useComponent(\"header\")\n .pipe(\n mountHeader({ document$, viewport$ }),\n shareReplay(1)\n )\n\n const main$ = useComponent(\"main\")\n .pipe(\n mountMain({ header$, viewport$ }),\n shareReplay(1)\n )\n\n /* ----------------------------------------------------------------------- */\n\n const navigation$ = useComponent(\"navigation\")\n .pipe(\n mountNavigation({ header$, main$, viewport$, screen$ }),\n shareReplay(1) // shareReplay because there might be late subscribers\n )\n\n const toc$ = useComponent(\"toc\")\n .pipe(\n mountTableOfContents({ header$, main$, viewport$, tablet$ }),\n shareReplay(1)\n )\n\n const tabs$ = useComponent(\"tabs\")\n .pipe(\n mountTabs({ header$, viewport$, screen$ }),\n shareReplay(1)\n )\n\n const hero$ = useComponent(\"hero\")\n .pipe(\n mountHero({ header$, viewport$ }),\n shareReplay(1)\n )\n\n /* ----------------------------------------------------------------------- */\n\n /* Search worker */\n const worker$ = defer(() => {\n const index = config.search && config.search.index\n ? config.search.index\n : undefined\n\n /* Fetch index if it wasn't passed explicitly */\n const index$ = typeof index !== \"undefined\"\n ? from(index)\n : base$\n .pipe(\n switchMap(base => ajax({\n url: `${base}/search/search_index.json`,\n responseType: \"json\",\n withCredentials: true\n })\n .pipe<SearchIndex>(\n pluck(\"response\")\n )\n )\n )\n\n return of(setupSearchWorker(config.search.worker, {\n base$, index$\n }))\n })\n\n /* ----------------------------------------------------------------------- */\n\n /* Mount search query */\n const search$ = worker$\n .pipe(\n switchMap(worker => {\n\n const query$ = useComponent(\"search-query\")\n .pipe(\n mountSearchQuery(worker, { transform: config.search.transform }),\n shareReplay(1)\n )\n\n /* Mount search reset */\n const reset$ = useComponent(\"search-reset\")\n .pipe(\n mountSearchReset(),\n shareReplay(1)\n )\n\n /* Mount search result */\n const result$ = useComponent(\"search-result\")\n .pipe(\n mountSearchResult(worker, { query$ }),\n shareReplay(1)\n )\n\n return useComponent(\"search\")\n .pipe(\n mountSearch(worker, { query$, reset$, result$ }),\n shareReplay(1)\n )\n }),\n catchError(() => {\n useComponent(\"search\")\n .subscribe(el => el.hidden = true) // TODO: Hack\n return NEVER\n })\n )\n\n /* ----------------------------------------------------------------------- */\n\n // // put into search...\n hash$\n .pipe(\n tap(() => setToggle(\"search\", false)),\n delay(125), // ensure that it runs after the body scroll reset...\n )\n .subscribe(hash => setLocationHash(`#${hash}`))\n\n // TODO: scroll restoration must be centralized\n combineLatest([\n watchToggle(\"search\"),\n tablet$,\n ])\n .pipe(\n withLatestFrom(viewport$),\n switchMap(([[toggle, tablet], { offset: { y }}]) => {\n const active = toggle && !tablet\n return document$\n .pipe(\n delay(active ? 400 : 100),\n observeOn(animationFrameScheduler),\n tap(({ body }) => active\n ? setScrollLock(body, y)\n : resetScrollLock(body)\n )\n )\n })\n )\n .subscribe()\n\n /* ----------------------------------------------------------------------- */\n\n /* Always close drawer on click */\n fromEvent<MouseEvent>(document.body, \"click\")\n .pipe(\n filter(ev => !(ev.metaKey || ev.ctrlKey)),\n filter(ev => {\n if (ev.target instanceof HTMLElement) {\n const el = ev.target.closest(\"a\") // TODO: abstract as link click?\n if (el && isLocalLocation(el)) {\n return true\n }\n }\n return false\n })\n )\n .subscribe(() => {\n setToggle(\"drawer\", false)\n })\n\n /* Enable instant loading, if not on file:// protocol */\n if (config.features.includes(\"instant\") && location.protocol !== \"file:\") {\n\n /* Fetch sitemap and extract URL whitelist */\n base$\n .pipe(\n switchMap(base => ajax({\n url: `${base}/sitemap.xml`,\n responseType: \"document\",\n withCredentials: true\n })\n .pipe<Document>(\n pluck(\"response\")\n )\n ),\n withLatestFrom(base$),\n map(([document, base]) => {\n const urls = getElements(\"loc\", document)\n .map(node => node.textContent!)\n\n // Hack: This is a temporary fix to normalize instant loading lookup\n // on localhost and Netlify previews. If this approach proves to be\n // suitable, we'll refactor URL whitelisting anyway. We take the two\n // shortest URLs and determine the common prefix to isolate the\n // domain. If there're no two domains, we just leave it as-is, as\n // there isn't anything to be loaded anway.\n if (urls.length > 1) {\n const [a, b] = sortBy(prop(\"length\"), urls)\n\n /* Determine common prefix */\n let index = 0\n if (a === b)\n index = a.length\n else\n while (a.charAt(index) === b.charAt(index))\n index++\n\n /* Replace common prefix (i.e. base) with effective base */\n for (let i = 0; i < urls.length; i++)\n urls[i] = urls[i].replace(a.slice(0, index), `${base}/`)\n }\n return urls\n })\n )\n .subscribe(urls => {\n setupInstantLoading(urls, { document$, location$, viewport$ })\n })\n }\n\n /* ----------------------------------------------------------------------- */\n\n /* Unhide permalinks on first tab */\n keyboard$\n .pipe(\n filter(key => key.mode === \"global\" && key.type === \"Tab\"),\n take(1)\n )\n .subscribe(() => {\n for (const link of getElements(\".headerlink\"))\n link.style.visibility = \"visible\"\n })\n\n /* ----------------------------------------------------------------------- */\n\n const state = {\n\n /* Browser observables */\n document$,\n location$,\n viewport$,\n\n /* Component observables */\n header$,\n hero$,\n main$,\n navigation$,\n search$,\n tabs$,\n toc$,\n\n /* Integration observables */\n clipboard$,\n keyboard$,\n dialog$\n }\n\n /* Subscribe to all observables */\n merge(...values(state))\n .subscribe()\n return state\n}\n","/*\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport { Observable, combineLatest } from \"rxjs\"\nimport { distinctUntilKeyChanged, map } from \"rxjs/operators\"\n\nimport { Viewport, getElements } from \"browser\"\n\n/* ----------------------------------------------------------------------------\n * Helper types\n * ------------------------------------------------------------------------- */\n\n/**\n * Mount options\n */\ninterface MountOptions {\n document$: Observable<Document> /* Document observable */\n viewport$: Observable<Viewport> /* Viewport observable */\n}\n\n/* ----------------------------------------------------------------------------\n * Functions\n * ------------------------------------------------------------------------- */\n\n/**\n * Patch all `code` elements\n *\n * This function will make overflowing code blocks focusable via keyboard, so\n * they can be scrolled without a mouse.\n *\n * @param options - Options\n */\nexport function patchCodeBlocks(\n { document$, viewport$ }: MountOptions\n): void {\n const els$ = document$\n .pipe(\n map(() => getElements<HTMLTableElement>(\"pre > code\"))\n )\n\n /* Observe viewport size only */\n const size$ = viewport$\n .pipe(\n distinctUntilKeyChanged(\"size\")\n )\n\n /* Make overflowing elements focusable */\n combineLatest([els$, size$])\n .subscribe(([els]) => {\n for (const el of els) {\n if (el.scrollWidth > el.clientWidth)\n el.setAttribute(\"tabindex\", \"0\")\n else\n el.removeAttribute(\"tabindex\")\n }\n })\n}\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { identity } from \"ramda\"\r\nimport { Observable, fromEvent, merge } from \"rxjs\"\r\nimport {\r\n filter,\r\n map,\r\n switchMapTo,\r\n tap\r\n} from \"rxjs/operators\"\r\n\r\nimport {\r\n getElement,\r\n getElements,\r\n watchMedia\r\n} from \"browser\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch options\r\n */\r\ninterface PatchOptions {\r\n document$: Observable<Document> /* Document observable */\r\n hash$: Observable<string> /* Location hash observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch all `details` elements\r\n *\r\n * This function will ensure that all `details` tags are opened prior to\r\n * printing, so the whole content of the page is included, and on anchor jumps.\r\n *\r\n * @param options - Options\r\n */\r\nexport function patchDetails(\r\n { document$, hash$ }: PatchOptions\r\n): void {\r\n const els$ = document$\r\n .pipe(\r\n map(() => getElements<HTMLDetailsElement>(\"details\"))\r\n )\r\n\r\n /* Open all details before printing */\r\n merge(\r\n watchMedia(\"print\").pipe(filter(identity)), /* Webkit */\r\n fromEvent(window, \"beforeprint\") /* IE, FF */\r\n )\r\n .pipe(\r\n switchMapTo(els$)\r\n )\r\n .subscribe(els => {\r\n for (const el of els)\r\n el.setAttribute(\"open\", \"\")\r\n })\r\n\r\n /* Open parent details and fix anchor jump */\r\n hash$\r\n .pipe(\r\n map(id => getElement(`[id=\"${id}\"]`)!),\r\n filter(el => typeof el !== \"undefined\"),\r\n tap(el => {\r\n const details = el.closest(\"details\")\r\n if (details && !details.open)\r\n details.setAttribute(\"open\", \"\")\r\n })\r\n )\r\n .subscribe(el => el.scrollIntoView())\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable } from \"rxjs\"\r\nimport { map, skip, withLatestFrom } from \"rxjs/operators\"\r\n\r\nimport {\r\n createElement,\r\n getElements,\r\n replaceElement\r\n} from \"browser\"\r\nimport { useComponent } from \"components\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch options\r\n */\r\ninterface PatchOptions {\r\n document$: Observable<Document> /* Document observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch all `script` elements\r\n *\r\n * This function must be run after a document switch, which means the first\r\n * emission must be ignored.\r\n *\r\n * @param options - Options\r\n */\r\nexport function patchScripts(\r\n { document$ }: PatchOptions\r\n): void {\r\n const els$ = document$\r\n .pipe(\r\n skip(1),\r\n withLatestFrom(useComponent(\"container\")),\r\n map(([, el]) => getElements<HTMLScriptElement>(\"script\", el))\r\n )\r\n\r\n /* Evaluate all scripts via replacement */\r\n els$.subscribe(els => {\r\n for (const el of els) {\r\n if (el.src || /(^|\\/javascript)$/i.test(el.type)) {\r\n const script = createElement(\"script\")\r\n const key = el.src ? \"src\" : \"textContent\"\r\n script[key] = el[key]!\r\n replaceElement(el, script)\r\n }\r\n }\r\n })\r\n}\r\n","/*\r\n * Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>\r\n *\r\n * Permission is hereby granted, free of charge, to any person obtaining a copy\r\n * of this software and associated documentation files (the \"Software\"), to\r\n * deal in the Software without restriction, including without limitation the\r\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\r\n * sell copies of the Software, and to permit persons to whom the Software is\r\n * furnished to do so, subject to the following conditions:\r\n *\r\n * The above copyright notice and this permission notice shall be included in\r\n * all copies or substantial portions of the Software.\r\n *\r\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\r\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\r\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\r\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\r\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\r\n * IN THE SOFTWARE.\r\n */\r\n\r\nimport { Observable } from \"rxjs\"\r\nimport { map } from \"rxjs/operators\"\r\n\r\nimport {\r\n createElement,\r\n getElements,\r\n replaceElement\r\n} from \"browser\"\r\nimport { renderTable } from \"templates\"\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Helper types\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Mount options\r\n */\r\ninterface MountOptions {\r\n document$: Observable<Document> /* Document observable */\r\n}\r\n\r\n/* ----------------------------------------------------------------------------\r\n * Functions\r\n * ------------------------------------------------------------------------- */\r\n\r\n/**\r\n * Patch all `table` elements\r\n *\r\n * This function will re-render all tables by wrapping them to improve overflow\r\n * scrolling on smaller screen sizes.\r\n *\r\n * @param options - Options\r\n */\r\nexport function patchTables(\r\n { document$ }: MountOptions\r\n): void {\r\n const sentinel = createElement(\"table\")\r\n document$\r\n .pipe(\r\n map(() => getElements<HTMLTableElement>(\"table:not([class])\"))\r\n )\r\n .subscribe(els => {\r\n for (const el of els) {\r\n replaceElement(el, sentinel)\r\n replaceElement(sentinel, renderTable(el))\r\n }\r\n })\r\n}\r\n"],"sourceRoot":""} |