mirror of
https://github.com/InsanusMokrassar/MicroUtils.git
synced 2025-01-13 03:09:56 +00:00
372 lines
13 KiB
JavaScript
372 lines
13 KiB
JavaScript
/*
|
|
* Copyright 2014-2024 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
|
|
*/
|
|
/** When Dokka is viewed via iframe, local storage could be inaccessible (see https://github.com/Kotlin/dokka/issues/3323)
|
|
* This is a wrapper around local storage to prevent errors in such cases
|
|
* */
|
|
const safeLocalStorage = (() => {
|
|
let isLocalStorageAvailable = false;
|
|
try {
|
|
const testKey = '__testLocalStorageKey__';
|
|
localStorage.setItem(testKey, testKey);
|
|
localStorage.removeItem(testKey);
|
|
isLocalStorageAvailable = true;
|
|
} catch (e) {
|
|
console.error('Local storage is not available', e);
|
|
}
|
|
|
|
return {
|
|
getItem: (key) => {
|
|
if (!isLocalStorageAvailable) {
|
|
return null;
|
|
}
|
|
return localStorage.getItem(key);
|
|
},
|
|
setItem: (key, value) => {
|
|
if (!isLocalStorageAvailable) {
|
|
return;
|
|
}
|
|
localStorage.setItem(key, value);
|
|
}
|
|
};
|
|
})();
|
|
|
|
filteringContext = {
|
|
dependencies: {},
|
|
restrictedDependencies: [],
|
|
activeFilters: []
|
|
}
|
|
let highlightedAnchor;
|
|
let topNavbarOffset;
|
|
let instances = [];
|
|
let sourcesetNotification;
|
|
|
|
const samplesDarkThemeName = 'darcula'
|
|
const samplesLightThemeName = 'idea'
|
|
|
|
window.addEventListener('load', () => {
|
|
document.querySelectorAll("div[data-platform-hinted]")
|
|
.forEach(elem => elem.addEventListener('click', (event) => togglePlatformDependent(event, elem)))
|
|
const filterSection = document.getElementById('filter-section')
|
|
if (filterSection) {
|
|
filterSection.addEventListener('click', (event) => filterButtonHandler(event))
|
|
initializeFiltering()
|
|
}
|
|
if (typeof initTabs === 'function') {
|
|
initTabs() // initTabs comes from ui-kit/tabs
|
|
}
|
|
handleAnchor()
|
|
topNavbarOffset = document.getElementById('navigation-wrapper')
|
|
darkModeSwitch()
|
|
})
|
|
|
|
const darkModeSwitch = () => {
|
|
const localStorageKey = "dokka-dark-mode"
|
|
const storage = safeLocalStorage.getItem(localStorageKey)
|
|
const osDarkSchemePreferred = window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches
|
|
const darkModeEnabled = storage ? JSON.parse(storage) : osDarkSchemePreferred
|
|
const element = document.getElementById("theme-toggle-button")
|
|
initPlayground(darkModeEnabled ? samplesDarkThemeName : samplesLightThemeName)
|
|
|
|
element.addEventListener('click', () => {
|
|
const enabledClasses = document.getElementsByTagName("html")[0].classList
|
|
enabledClasses.toggle("theme-dark")
|
|
|
|
//if previously we had saved dark theme then we set it to light as this is what we save in local storage
|
|
const darkModeEnabled = enabledClasses.contains("theme-dark")
|
|
if (darkModeEnabled) {
|
|
initPlayground(samplesDarkThemeName)
|
|
} else {
|
|
initPlayground(samplesLightThemeName)
|
|
}
|
|
safeLocalStorage.setItem(localStorageKey, JSON.stringify(darkModeEnabled))
|
|
})
|
|
}
|
|
|
|
const initPlayground = (theme) => {
|
|
if (!samplesAreEnabled()) return
|
|
instances.forEach(instance => instance.destroy())
|
|
instances = []
|
|
|
|
// Manually tag code fragments as not processed by playground since we also manually destroy all of its instances
|
|
document.querySelectorAll('code.runnablesample').forEach(node => {
|
|
node.removeAttribute("data-kotlin-playground-initialized");
|
|
})
|
|
|
|
KotlinPlayground('code.runnablesample', {
|
|
getInstance: playgroundInstance => {
|
|
instances.push(playgroundInstance)
|
|
},
|
|
theme: theme
|
|
});
|
|
}
|
|
|
|
// We check if type is accessible from the current scope to determine if samples script is present
|
|
// As an alternative we could extract this samples-specific script to new js file but then we would handle dark mode in 2 separate files which is not ideal
|
|
const samplesAreEnabled = () => {
|
|
try {
|
|
if (typeof KotlinPlayground === 'undefined') {
|
|
// KotlinPlayground is exported universally as a global variable or as a module
|
|
// Due to possible interaction with other js scripts KotlinPlayground may not be accessible directly from `window`, so we need an additional check
|
|
KotlinPlayground = exports.KotlinPlayground;
|
|
}
|
|
return typeof KotlinPlayground === 'function';
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
// Hash change is needed in order to allow for linking inside the same page with anchors
|
|
// If this is not present user is forced to refresh the site in order to use an anchor
|
|
window.onhashchange = handleAnchor
|
|
|
|
function scrollToElementInContent(element) {
|
|
const scrollToElement = () => document.getElementById('main').scrollTo({
|
|
top: element.offsetTop - topNavbarOffset.offsetHeight,
|
|
behavior: "smooth"
|
|
})
|
|
|
|
const waitAndScroll = () => {
|
|
setTimeout(() => {
|
|
if (topNavbarOffset) {
|
|
scrollToElement()
|
|
} else {
|
|
waitForScroll()
|
|
}
|
|
}, 50)
|
|
}
|
|
|
|
if (topNavbarOffset) {
|
|
scrollToElement()
|
|
} else {
|
|
waitAndScroll()
|
|
}
|
|
}
|
|
|
|
|
|
function handleAnchor() {
|
|
if (highlightedAnchor) {
|
|
highlightedAnchor.classList.remove('anchor-highlight')
|
|
highlightedAnchor = null;
|
|
}
|
|
|
|
let searchForContentTarget = function (element) {
|
|
if (element && element.hasAttribute) {
|
|
if (element.hasAttribute("data-togglable")) return element.getAttribute("data-togglable");
|
|
else return searchForContentTarget(element.parentNode)
|
|
} else return null
|
|
}
|
|
|
|
let findAnyTab = function (target) {
|
|
let result = null
|
|
document.querySelectorAll('div[tabs-section] > button[data-togglable]')
|
|
.forEach(node => {
|
|
if(node.getAttribute("data-togglable").split(",").includes(target)) {
|
|
result = node
|
|
}
|
|
})
|
|
return result
|
|
}
|
|
|
|
let anchor = window.location.hash
|
|
if (anchor !== "") {
|
|
anchor = anchor.substring(1)
|
|
let element = document.querySelector('a[data-name="' + anchor + '"]')
|
|
|
|
if (element) {
|
|
const content = element.nextElementSibling
|
|
const contentStyle = window.getComputedStyle(content)
|
|
if(contentStyle.display === 'none') {
|
|
let tab = findAnyTab(searchForContentTarget(content))
|
|
if (tab) {
|
|
toggleSections(tab) // toggleSections comes from ui-kit/tabs
|
|
}
|
|
}
|
|
|
|
if (content) {
|
|
content.classList.add('anchor-highlight')
|
|
highlightedAnchor = content
|
|
}
|
|
|
|
scrollToElementInContent(element)
|
|
}
|
|
}
|
|
}
|
|
|
|
function filterButtonHandler(event) {
|
|
if (event.target.tagName === "BUTTON" && event.target.hasAttribute("data-filter")) {
|
|
let sourceset = event.target.getAttribute("data-filter")
|
|
if (filteringContext.activeFilters.indexOf(sourceset) !== -1) {
|
|
filterSourceset(sourceset)
|
|
} else {
|
|
unfilterSourceset(sourceset)
|
|
}
|
|
}
|
|
}
|
|
|
|
function initializeFiltering() {
|
|
filteringContext.dependencies = JSON.parse(sourceset_dependencies)
|
|
document.querySelectorAll("#filter-section > button")
|
|
.forEach(p => filteringContext.restrictedDependencies.push(p.getAttribute("data-filter")))
|
|
Object.keys(filteringContext.dependencies).forEach(p => {
|
|
filteringContext.dependencies[p] = filteringContext.dependencies[p]
|
|
.filter(q => -1 !== filteringContext.restrictedDependencies.indexOf(q))
|
|
})
|
|
let cached = safeLocalStorage.getItem('inactive-filters')
|
|
if (cached) {
|
|
let parsed = JSON.parse(cached)
|
|
filteringContext.activeFilters = filteringContext.restrictedDependencies
|
|
.filter(q => parsed.indexOf(q) === -1)
|
|
} else {
|
|
filteringContext.activeFilters = filteringContext.restrictedDependencies
|
|
}
|
|
refreshFiltering()
|
|
}
|
|
|
|
function filterSourceset(sourceset) {
|
|
filteringContext.activeFilters = filteringContext.activeFilters.filter(p => p !== sourceset)
|
|
refreshFiltering()
|
|
addSourcesetFilterToCache(sourceset)
|
|
}
|
|
|
|
function unfilterSourceset(sourceset) {
|
|
if (filteringContext.activeFilters.length === 0) {
|
|
filteringContext.activeFilters = filteringContext.dependencies[sourceset].concat([sourceset])
|
|
refreshFiltering()
|
|
filteringContext.dependencies[sourceset].concat([sourceset]).forEach(p => removeSourcesetFilterFromCache(p))
|
|
} else {
|
|
filteringContext.activeFilters.push(sourceset)
|
|
refreshFiltering()
|
|
removeSourcesetFilterFromCache(sourceset)
|
|
}
|
|
|
|
}
|
|
|
|
function addSourcesetFilterToCache(sourceset) {
|
|
let cached = safeLocalStorage.getItem('inactive-filters')
|
|
if (cached) {
|
|
let parsed = JSON.parse(cached)
|
|
safeLocalStorage.setItem('inactive-filters', JSON.stringify(parsed.concat([sourceset])))
|
|
} else {
|
|
safeLocalStorage.setItem('inactive-filters', JSON.stringify([sourceset]))
|
|
}
|
|
}
|
|
|
|
function removeSourcesetFilterFromCache(sourceset) {
|
|
let cached = safeLocalStorage.getItem('inactive-filters')
|
|
if (cached) {
|
|
let parsed = JSON.parse(cached)
|
|
safeLocalStorage.setItem('inactive-filters', JSON.stringify(parsed.filter(p => p !== sourceset)))
|
|
}
|
|
}
|
|
|
|
function refreshSourcesetsCache() {
|
|
safeLocalStorage.setItem('inactive-filters', JSON.stringify(filteringContext.restrictedDependencies.filter(p => -1 === filteringContext.activeFilters.indexOf(p))))
|
|
}
|
|
|
|
|
|
function togglePlatformDependent(e, container) {
|
|
let target = e.target
|
|
if (target.tagName !== 'BUTTON') return;
|
|
let index = target.getAttribute('data-toggle')
|
|
|
|
for (let child of container.children) {
|
|
if (child.hasAttribute('data-toggle-list')) {
|
|
for (let bm of child.children) {
|
|
if (bm === target) {
|
|
bm.setAttribute('data-active', "")
|
|
} else if (bm !== target) {
|
|
bm.removeAttribute('data-active')
|
|
}
|
|
}
|
|
} else if (child.getAttribute('data-togglable') === index) {
|
|
child.setAttribute('data-active', "")
|
|
} else {
|
|
child.removeAttribute('data-active')
|
|
}
|
|
}
|
|
}
|
|
|
|
function refreshFiltering() {
|
|
let sourcesetList = filteringContext.activeFilters
|
|
document.querySelectorAll("[data-filterable-set]")
|
|
.forEach(
|
|
elem => {
|
|
let platformList = elem.getAttribute("data-filterable-set").split(',').filter(v => -1 !== sourcesetList.indexOf(v))
|
|
elem.setAttribute("data-filterable-current", platformList.join(','))
|
|
}
|
|
)
|
|
refreshFilterButtons()
|
|
refreshPlatformTabs()
|
|
refreshNoContentNotification()
|
|
refreshPlaygroundSamples()
|
|
}
|
|
|
|
function refreshPlaygroundSamples() {
|
|
document.querySelectorAll('code.runnablesample').forEach(node => {
|
|
const playground = node.KotlinPlayground;
|
|
/* Some samples may be hidden by filter, they have 0px height for visible code area
|
|
* after rendering. Call this method for re-calculate code area height */
|
|
playground && playground.view.codemirror.refresh();
|
|
});
|
|
}
|
|
|
|
function refreshNoContentNotification() {
|
|
const element = document.getElementsByClassName("main-content")[0]
|
|
const filteredMessage = document.querySelector(".filtered-message")
|
|
|
|
if(filteringContext.activeFilters.length === 0){
|
|
element.style.display = "none";
|
|
|
|
if (!filteredMessage) {
|
|
const appended = document.createElement("div")
|
|
appended.className = "filtered-message"
|
|
appended.innerText = "All documentation is filtered, please adjust your source set filters in top-right corner of the screen"
|
|
sourcesetNotification = appended
|
|
element.parentNode.prepend(appended)
|
|
}
|
|
} else {
|
|
if(sourcesetNotification) sourcesetNotification.remove()
|
|
element.style.display = "block"
|
|
}
|
|
}
|
|
|
|
function refreshPlatformTabs() {
|
|
document.querySelectorAll(".platform-hinted > .platform-bookmarks-row").forEach(
|
|
p => {
|
|
let active = false;
|
|
let firstAvailable = null
|
|
p.childNodes.forEach(
|
|
element => {
|
|
if (element.getAttribute("data-filterable-current") !== '') {
|
|
if (firstAvailable === null) {
|
|
firstAvailable = element
|
|
}
|
|
if (element.hasAttribute("data-active")) {
|
|
active = true;
|
|
}
|
|
}
|
|
}
|
|
)
|
|
if (active === false && firstAvailable) {
|
|
firstAvailable.click()
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
function refreshFilterButtons() {
|
|
document.querySelectorAll("#filter-section > button")
|
|
.forEach(f => {
|
|
if (filteringContext.activeFilters.indexOf(f.getAttribute("data-filter")) !== -1) {
|
|
f.setAttribute("data-active", "")
|
|
} else {
|
|
f.removeAttribute("data-active")
|
|
}
|
|
})
|
|
document.querySelectorAll("#filter-section .checkbox--input")
|
|
.forEach(f => {
|
|
f.checked = filteringContext.activeFilters.indexOf(f.getAttribute("data-filter")) !== -1;
|
|
})
|
|
}
|