mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-05 05:09:03 +00:00
194 lines
5.6 KiB
JavaScript
194 lines
5.6 KiB
JavaScript
/*
|
|
* Documentative
|
|
* (c) 2020 dragonwocky <thedragonring.bod@gmail.com>
|
|
* (https://dragonwocky.me/) under the MIT license
|
|
*/
|
|
|
|
class Scrollnav {
|
|
constructor(menu, content, options) {
|
|
if (!(menu instanceof HTMLElement))
|
|
throw Error('scrollnav: invalid <menu> element provided');
|
|
if (!(content instanceof HTMLElement))
|
|
throw Error('scrollnav: invalid <content> element provided');
|
|
if (typeof options !== 'object') options = {};
|
|
|
|
if (Scrollnav.prototype.INITIATED)
|
|
throw Error('scrollnav: only 1 instance per page allowed!');
|
|
Scrollnav.prototype.INITIATED = true;
|
|
|
|
this.ID;
|
|
this.ticking = [];
|
|
this._menu = menu;
|
|
this._content = content;
|
|
this._links = [];
|
|
this._sections = [...this._menu.querySelectorAll('ul li a')].reduce(
|
|
(list, link) => {
|
|
if (!link.getAttribute('href').startsWith('#')) return list;
|
|
let section = this._content.querySelector(link.getAttribute('href'));
|
|
if (!section) return list;
|
|
|
|
this._links.push(link);
|
|
link.onclick = async ev => {
|
|
ev.preventDefault();
|
|
const ID = link.getAttribute('href');
|
|
this.highlightHeading(ID);
|
|
this.scrollContent(ID);
|
|
this.setHash(ID);
|
|
};
|
|
|
|
return [...list, section];
|
|
},
|
|
[]
|
|
);
|
|
this._topheading = `#${this._sections[0].id}`;
|
|
|
|
window.onhashchange = this.watchHash.bind(this);
|
|
this._content.addEventListener('scroll', ev => {
|
|
if (!this.ticking.length) {
|
|
this.ticking.push(1);
|
|
requestAnimationFrame(() => {
|
|
this.watchScroll(ev);
|
|
this.ticking.pop();
|
|
});
|
|
}
|
|
});
|
|
|
|
this.set(null, false);
|
|
return this;
|
|
}
|
|
|
|
set(ID, smooth) {
|
|
this.highlightHeading(ID);
|
|
this.scrollMenu(ID, smooth);
|
|
this.scrollContent(ID, smooth);
|
|
this.setHash(ID);
|
|
}
|
|
|
|
parseID(ID) {
|
|
if (!ID || typeof ID !== 'string') ID = location.hash || this._topheading;
|
|
if (!ID.startsWith('#')) ID = `#${ID}`;
|
|
if (!this._links.find(el => el.getAttribute('href') === ID))
|
|
ID = this._topheading;
|
|
this.ID = ID;
|
|
return ID;
|
|
}
|
|
highlightHeading(ID) {
|
|
this.parseID(ID);
|
|
this._links.forEach(el =>
|
|
el.getAttribute('href') === this.ID
|
|
? el.classList.add('active')
|
|
: el.classList.remove('active')
|
|
);
|
|
return true;
|
|
}
|
|
|
|
watchHash(ev) {
|
|
ev.preventDefault();
|
|
if (ev.newURL !== ev.oldURL) {
|
|
this.set();
|
|
}
|
|
}
|
|
setHash(ID) {
|
|
if (!history.replaceState) return false;
|
|
this.parseID(ID);
|
|
history.replaceState(null, null, ID === this._topheading ? '#' : this.ID);
|
|
return true;
|
|
}
|
|
|
|
scrollContent(ID, smooth = true) {
|
|
this.ticking.push(1);
|
|
this.parseID(ID);
|
|
let offset = this._sections.find(el => `#${el.id}` === this.ID).offsetTop;
|
|
if (offset < this._content.clientHeight / 2) offset = 0;
|
|
this._content.scroll({
|
|
top: offset,
|
|
behavior: smooth ? 'smooth' : 'auto'
|
|
});
|
|
setTimeout(() => this.ticking.pop(), 1000);
|
|
return true;
|
|
}
|
|
scrollMenu(ID, smooth = true) {
|
|
this.parseID(ID);
|
|
let offset = this._links.find(el => el.getAttribute('href') === this.ID)
|
|
.offsetTop;
|
|
if (offset < this._menu.clientHeight / 2) offset = 0;
|
|
this._menu.scroll({
|
|
top: offset,
|
|
behavior: smooth ? 'smooth' : 'auto'
|
|
});
|
|
return true;
|
|
}
|
|
watchScroll(ev) {
|
|
const viewport = this._content.clientHeight,
|
|
ID = this._sections.reduce(
|
|
(carry, el) => {
|
|
const rect = el.getBoundingClientRect(),
|
|
height = rect.bottom - rect.top,
|
|
visible = {
|
|
top: rect.top >= 0 && rect.top < viewport,
|
|
bottom: rect.bottom > 0 && rect.top < viewport
|
|
};
|
|
let pixels = 0;
|
|
if (visible.top && visible.bottom) {
|
|
pixels = height; // whole el
|
|
} else if (visible.top) {
|
|
pixels = viewport - rect.top;
|
|
} else if (visible.bottom) {
|
|
pixels = rect.bottom;
|
|
} else if (height > viewport && rect.top < 0) {
|
|
const absolute = Math.abs(rect.top);
|
|
if (absolute < height) pixels = height - absolute; // part of el
|
|
}
|
|
pixels = (pixels / height) * 100;
|
|
return pixels > carry[0] ? [pixels, el] : carry;
|
|
},
|
|
[0, null]
|
|
)[1].id;
|
|
this.ID = ID;
|
|
this.scrollMenu(this.ID);
|
|
clearTimeout(this.afterScroll);
|
|
this.afterScroll = setTimeout(
|
|
() => void (this.highlightHeading(this.ID) && this.setHash(this.ID)),
|
|
100
|
|
);
|
|
}
|
|
}
|
|
|
|
let constructed = false;
|
|
const construct = () => {
|
|
if (document.readyState !== 'complete' || constructed) return false;
|
|
constructed = true;
|
|
|
|
if (
|
|
location.pathname.endsWith('index.html') &&
|
|
window.location.protocol === 'https:'
|
|
)
|
|
location.replace('./' + location.hash);
|
|
|
|
new Scrollnav(
|
|
document.querySelector('aside'),
|
|
document.querySelector('.documentative')
|
|
);
|
|
|
|
document.querySelector('.toggle button').onclick = () =>
|
|
document.body.classList.toggle('mobilemenu');
|
|
|
|
if (window.matchMedia) {
|
|
let prev;
|
|
const links = [...document.head.querySelectorAll('link[rel*="icon"]')],
|
|
pointer = document.createElement('link');
|
|
pointer.setAttribute('rel', 'icon');
|
|
document.head.appendChild(pointer);
|
|
setInterval(() => {
|
|
const match = links.find(link => window.matchMedia(link.media).matches);
|
|
if (!match || match.media === prev) return;
|
|
prev = match.media;
|
|
pointer.setAttribute('href', match.getAttribute('href'));
|
|
}, 500);
|
|
links.forEach(link => document.head.removeChild(link));
|
|
}
|
|
};
|
|
|
|
construct();
|
|
document.addEventListener('readystatechange', construct);
|