update cloudhill mods (#600)

This commit is contained in:
Ryo Hilmawan 2021-10-30 19:06:10 +07:00 committed by GitHub
parent 3e7319464e
commit 0a64c4eb5c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 338 additions and 214 deletions

View File

@ -5,7 +5,7 @@
* under the MIT license * under the MIT license
*/ */
.notion-code-block.line-numbers > div { .notion-code-block.line-numbers {
position: relative; position: relative;
} }
@ -25,3 +25,6 @@
overflow: hidden; overflow: hidden;
pointer-events: none; pointer-events: none;
} }
#code-line-numbers:empty {
display: none;
}

View File

@ -14,7 +14,7 @@ module.exports = {
tags: ['extension'], tags: ['extension'],
name: 'code line numbers', name: 'code line numbers',
desc: 'adds line numbers to code blocks.', desc: 'adds line numbers to code blocks.',
version: '1.1.1', version: '1.2.0',
author: 'CloudHill', author: 'CloudHill',
options: [ options: [
{ {
@ -44,23 +44,18 @@ module.exports = {
function handle(list) { function handle(list) {
queue = []; queue = [];
for (let { addedNodes } of list) { for (let { addedNodes, target } of list) {
if ( const block = target.querySelector('.line-numbers.notion-code-block') ||
addedNodes[0] &&
( (
addedNodes[0].className === 'notion-page-content' || addedNodes[0]?.classList?.contains('.notion-code-block') &&
( addedNodes[0].querySelector('.line-numbers.notion-code-block')
addedNodes[0].querySelector && );
addedNodes[0].querySelector('.notion-code-block.line-numbers')
) if (block) {
) if (block.dataset.numbered) return;
) {
resizeObserver.disconnect();
const codeBlocks = document.querySelectorAll('.notion-code-block.line-numbers');
codeBlocks.forEach(block => {
number(block); number(block);
block.dataset.numbered = true;
resizeObserver.observe(block); resizeObserver.observe(block);
});
} }
} }
} }
@ -74,12 +69,14 @@ module.exports = {
'<span id="code-line-numbers"></span>' '<span id="code-line-numbers"></span>'
); );
// set size
const blockStyle = window.getComputedStyle(block.children[0]); const blockStyle = window.getComputedStyle(block.children[0]);
numbers.style.top = blockStyle.paddingTop; numbers.style.top = blockStyle.paddingTop;
numbers.style.bottom = blockStyle.paddingBottom; numbers.style.bottom = blockStyle.paddingBottom;
block.append(numbers); block.append(numbers);
// get lineHeight
const temp = createElement('<span>A</span>'); const temp = createElement('<span>A</span>');
block.firstChild.append(temp); block.firstChild.append(temp);
block.lineHeight = temp.getBoundingClientRect().height; block.lineHeight = temp.getBoundingClientRect().height;
@ -109,7 +106,7 @@ module.exports = {
if (store().single_lined || codeLineNumbers.length > 2) { if (store().single_lined || codeLineNumbers.length > 2) {
block.firstChild.classList.add('code-numbered'); block.firstChild.classList.add('code-numbered');
numbers.innerText = codeLineNumbers; numbers.innerText = codeLineNumbers || 1;
} else { } else {
block.firstChild.classList.remove('code-numbered'); block.firstChild.classList.remove('code-numbered');
numbers.innerText = ''; numbers.innerText = '';

View File

@ -14,7 +14,7 @@ module.exports = {
tags: ['extension'], tags: ['extension'],
name: 'collapsible headers', name: 'collapsible headers',
desc: 'adds toggles to collapse header sections.', desc: 'adds toggles to collapse header sections.',
version: '1.0.0', version: '1.0.1',
author: 'CloudHill', author: 'CloudHill',
options: [ options: [
{ {
@ -92,15 +92,18 @@ module.exports = {
for (let { addedNodes } of list) { for (let { addedNodes } of list) {
if ( if (
addedNodes[0] && addedNodes[0] &&
addedNodes[0].className === 'notion-page-content' addedNodes[0].className === 'notion-presence-container'
) { ) {
showSelectedHeader(); showSelectedHeader();
initHeaderToggles(); initHeaderToggles();
contentObserver.disconnect(); contentObserver.disconnect();
contentObserver.observe(addedNodes[0], { contentObserver.observe(
document.querySelector('.notion-page-content'),
{
childList: true, childList: true,
subtree: true, subtree: true,
}); }
);
} }
} }
} }

View File

@ -14,7 +14,7 @@ module.exports = {
tags: ['extension', 'panel'], tags: ['extension', 'panel'],
name: 'outliner', name: 'outliner',
desc: 'table of contents.', desc: 'table of contents.',
version: '1.2.1', version: '1.2.2',
author: 'CloudHill', author: 'CloudHill',
options: [ options: [
{ {

View File

@ -16,7 +16,7 @@ module.exports = (store, __exports) => {
const pageObserver = new MutationObserver((list, observer) => { const pageObserver = new MutationObserver((list, observer) => {
for ( let { addedNodes } of list) { for ( let { addedNodes } of list) {
if (addedNodes[0]) { if (addedNodes[0]) {
if (addedNodes[0].className === 'notion-page-content') { if (addedNodes[0].className === 'notion-presence-container') {
startContentObserver(); startContentObserver();
} }
// Clear outline on database pages // Clear outline on database pages
@ -147,7 +147,7 @@ module.exports = (store, __exports) => {
// Find headers when switching panels // Find headers when switching panels
if (document.querySelector('.notion-page-content')) { if (document.querySelector('.notion-page-content')) {
startContentObserver(); startContentObserver();
}; }
pageObserver.observe(document.body, { pageObserver.observe(document.body, {
childList: true, childList: true,
subtree: true, subtree: true,

View File

@ -36,7 +36,7 @@
font-size: 20px; font-size: 20px;
background: var(--theme--interactive_hover); background: var(--theme--interactive_hover);
box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border); box-shadow: var(--theme--box-shadow);
} }
.notion-scroll-button { .notion-scroll-button {
@ -55,6 +55,6 @@
} }
.notion-scroll-button > svg { .notion-scroll-button > svg {
width: 18px; width: 16px;
height: 18px; height: 16px;
} }

View File

@ -1 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="currentColor" viewBox="0 0 250 250"><path fill="currentColor" d="M102.683 234.52V110.968H67.5474L124.859 12.184L182.459 110.968H147.323V234.52H102.683Z"/></svg> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 14 14"><polygon points="7,4.7 1.1,10.6 0,9.4 6.5,2.9 7,2.4 7.5,2.9 14,9.4 12.9,10.6"/></svg>

Before

Width:  |  Height:  |  Size: 231 B

After

Width:  |  Height:  |  Size: 145 B

View File

@ -17,7 +17,7 @@ module.exports = {
name: 'scroll to top', name: 'scroll to top',
desc: desc:
'add an arrow above the help button to scroll back to the top of a page.', 'add an arrow above the help button to scroll back to the top of a page.',
version: '1.0.0', version: '1.1.0',
author: 'CloudHill', author: 'CloudHill',
options: [ options: [
{ {
@ -97,7 +97,7 @@ module.exports = {
for (let { addedNodes } of list) { for (let { addedNodes } of list) {
if ( if (
addedNodes[0] && addedNodes[0] &&
(addedNodes[0].className === 'notion-page-content' || (addedNodes[0].className === 'notion-presence-container' ||
addedNodes[0].className === 'notion-scroller') && addedNodes[0].className === 'notion-scroller') &&
top > 0 top > 0
) { ) {

View File

@ -5,17 +5,14 @@
* under the MIT license * under the MIT license
*/ */
.notion-frame { .notion-frame, .notion-cursor-listener > div[style*="flex-end"] {
transition: padding-right 300ms ease-in-out; transition: padding-right 300ms ease-in-out;
} }
.enhancer-panel--container { .enhancer-panel--container {
flex-grow: 0; flex-grow: 0;
flex-shrink: 0; flex-shrink: 0;
position: absolute; z-index: 999;
top: 0;
bottom: 0;
right: 0;
z-index: 99;
height: 100vh; height: 100vh;
background: var(--theme--sidebar); background: var(--theme--sidebar);
color: var(--theme--text_ui); color: var(--theme--text_ui);
@ -32,6 +29,7 @@
background: var(--theme--sidebar); background: var(--theme--sidebar);
cursor: auto; cursor: auto;
max-height: 100%; max-height: 100%;
z-index: 1;
transition: transform 300ms ease-in-out, transition: transform 300ms ease-in-out,
opacity 300ms ease-in-out, opacity 300ms ease-in-out,
right 300ms ease-in-out; right 300ms ease-in-out;
@ -64,6 +62,7 @@
} }
.enhancer-panel--header:hover { .enhancer-panel--header:hover {
background: var(--theme--interactive_hover); background: var(--theme--interactive_hover);
box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border);
} }
.enhancer-panel--icon { .enhancer-panel--icon {
@ -180,6 +179,16 @@
width: 6px; width: 6px;
} }
.enhancer-panel--hover-handle {
position: absolute;
right: 8px;
top: 60px;
height: 100%;
width: 20px;
}
/* overlay */
.enhancer-panel--overlay-container { .enhancer-panel--overlay-container {
position: fixed; position: fixed;
top: 0; top: 0;

View File

@ -16,11 +16,11 @@ module.exports = {
tags: ['extension', 'panel'], tags: ['extension', 'panel'],
name: 'side panel', name: 'side panel',
desc: 'adds a side panel to notion.', desc: 'adds a side panel to notion.',
version: '1.1.0', version: '1.2.1',
author: 'CloudHill', author: 'CloudHill',
hacks: { hacks: {
'renderer/preload.js'(store, __exports) { 'renderer/preload.js'(store, __exports) {
// Load icons // load icons
let icons = {}; let icons = {};
(async () => { (async () => {
icons.doubleChevron = await fs.readFile( path.resolve(__dirname, 'icons/double-chevron.svg') ); icons.doubleChevron = await fs.readFile( path.resolve(__dirname, 'icons/double-chevron.svg') );
@ -28,47 +28,53 @@ module.exports = {
icons.reload = await fs.readFile( path.resolve(__dirname, 'icons/reload.svg') ); icons.reload = await fs.readFile( path.resolve(__dirname, 'icons/reload.svg') );
})(); })();
// Load panel mods // load panel mods
let panelMods = let panelMods =
getEnhancements().loaded.filter( getEnhancements().loaded.filter(
mod => (mod.panel && (store('mods')[mod.id] || {}).enabled) mod => (mod.panel && (store('mods')[mod.id] || {}).enabled)
); );
// initialize panel values
panelMods.forEach(mod => initMod(mod)); panelMods.forEach(mod => initMod(mod));
// panelMods is an array containing objects with info about each panel
document.addEventListener('readystatechange', (event) => { document.addEventListener('readystatechange', (event) => {
if (document.readyState !== 'complete') return false; if (document.readyState !== 'complete') return false;
// if no panel mods activated
if (panelMods.length < 1) return; if (panelMods.length < 1) return;
const attempt_interval = setInterval(enhance, 500); const attempt_interval = setInterval(enhance, 500);
function enhance() { function enhance() {
// default panel width
if (!store().width) store().width = 220; if (!store().width) store().width = 220;
let curPanel = {}; let curPanel = {};
const frame = document.querySelector('.notion-frame'); if (!document.querySelector('.notion-sidebar')) return;
if (!frame) return;
clearInterval(attempt_interval); clearInterval(attempt_interval);
// Initialize panel // notion elements to manipulate
const frame = document.querySelector('.notion-frame');
const notionSidebarContainer = document
.querySelector(".notion-cursor-listener > div[style*=\"flex-end\"]");
// INITIALIZE PANEL
const container = createElement( const container = createElement(
'<div class="enhancer-panel--container"></div>' '<div class="enhancer-panel--container"></div>'
); );
const panel = createElement( const panel = createElement(
`<div id="enhancer-panel"></div>` `<div id="enhancer-panel"></div>`
); );
frame.after(container);
container.appendChild(panel); container.appendChild(panel);
// Panel contents // panel contents
const header = createElement(` const header = createElement(`
<div class="enhancer-panel--header"> <div class="enhancer-panel--header">
<div class="enhancer-panel--icon"></div> <div class="enhancer-panel--icon"></div>
<div class="enhancer-panel--title"></div> <div class="enhancer-panel--title"></div>
</div> </div>
`); `);
const toggle = createElement(
`<div class="enhancer-panel--toggle">${icons.doubleChevron}</div>`
);
const content = createElement( const content = createElement(
'<div id="enhancer-panel--content"></div>' '<div id="enhancer-panel--content"></div>'
); );
@ -77,31 +83,37 @@ module.exports = {
<div style="cursor: col-resize;"></div> <div style="cursor: col-resize;"></div>
</div> </div>
`); `);
panel.append(header, content, resize); panel.append(header, content, resize);
// Add switcher if there is more than one panel mods // add switcher if there is more than one panel mods
if (panelMods.length > 1) { if (panelMods.length > 1) {
header.addEventListener('click', renderSwitcher);
const switcherIcon = createElement( const switcherIcon = createElement(
`<div class="enhancer-panel--switcher-icon">${icons.switcher}</div>` `<div class="enhancer-panel--switcher-icon">${icons.switcher}</div>`
) )
header.appendChild(switcherIcon); header.appendChild(switcherIcon);
header.addEventListener('click', renderSwitcher);
} else { } else {
header.addEventListener('click', togglePanel); header.addEventListener('click', togglePanel);
} }
// add panel lock toggle
const toggle = createElement(
`<div class="enhancer-panel--toggle">${icons.doubleChevron}</div>`
);
header.appendChild(toggle); header.appendChild(toggle);
toggle.addEventListener('click', togglePanel); toggle.addEventListener('click', togglePanel);
// Keybind // render panel
notionSidebarContainer.after(container);
// toggle panel keybind
document.addEventListener('keyup', e => { document.addEventListener('keyup', e => {
const hotkey = { const hotkey = {
code: 'Backslash', code: 'Backslash',
ctrlKey: true, ctrlKey: true,
shiftKey: true, shiftKey: false,
metaKey: false, metaKey: false,
altKey: false, altKey: true,
}; };
for (let prop in hotkey) for (let prop in hotkey)
if (hotkey[prop] !== e[prop]) return; if (hotkey[prop] !== e[prop]) return;
@ -114,26 +126,34 @@ module.exports = {
enableResize(); enableResize();
// Attempt to load last opened mod // attempt to load last opened mod
let loaded = false; let loaded = false;
if (store().last_open) { if (store().last_open) {
panelMods.forEach(mod => { panelMods.forEach(mod => {
if (mod.id === store().last_open) { if (mod.id === store().last_open) {
loadContent(mod); loadPanelMod(mod);
loaded = true; loaded = true;
} }
}); });
} }
if (!loaded) loadContent(panelMods[0]); if (!loaded) loadPanelMod(panelMods[0]);
function loadContent(mod) {
if (curPanel.js && curPanel.js.onSwitch) curPanel.js.onSwitch(); // loads a panel mod
function loadPanelMod(mod) {
// call previous panel's onSwitch function
if (curPanel.js?.onSwitch) curPanel.js.onSwitch();
// set current panel
curPanel = mod.panel; curPanel = mod.panel;
store().last_open = mod.id; store().last_open = mod.id;
panel.querySelector('.enhancer-panel--title').innerHTML = mod.panel.name || mod.name;
// Reload button setPanelTitle(mod.panel.name);
setPanelIcon(mod.panel.icon);
setPanelContent(mod.panel.html);
panel.dataset.fullHeight = mod.panel.fullHeight || false;
// reload button
let reloadButton = panel.querySelector('.enhancer-panel--reload-button'); let reloadButton = panel.querySelector('.enhancer-panel--reload-button');
if (reloadButton) reloadButton.remove(); if (reloadButton) reloadButton.remove();
if (mod.panel.reload) { if (mod.panel.reload) {
@ -142,102 +162,168 @@ module.exports = {
) )
reloadButton.addEventListener('click', e => { reloadButton.addEventListener('click', e => {
e.stopPropagation(); e.stopPropagation();
loadContent(mod); loadPanelMod(mod);
}) })
panel.querySelector('.enhancer-panel--title').after(reloadButton); panel.querySelector('.enhancer-panel--title').after(reloadButton);
} }
panel.querySelector('.enhancer-panel--icon').innerHTML = mod.panel.icon; // execute panel's onLoad function
document.getElementById('enhancer-panel--content').innerHTML = mod.panel.html; if (curPanel.js?.onLoad) curPanel.js.onLoad();
panel.dataset.fullHeight = mod.panel.fullHeight || false; }
if (curPanel.js && curPanel.js.onLoad) function setPanelTitle(title) {
curPanel.js.onLoad(); panel.querySelector('.enhancer-panel--title').innerHTML = title;
}
function setPanelIcon(icon) {
panel.querySelector('.enhancer-panel--icon').innerHTML = icon;
}
function setPanelContent(content) {
document.getElementById('enhancer-panel--content').innerHTML = content;
}
function setPanelWidth(width) {
// update width
store().width = width;
panel.style.width = width + 'px';
if (isLocked()) {
// panel container width
container.style.width = width + 'px';
// adjust notion elements to make space on the right
frame.style.paddingRight = width + 'px';
notionSidebarContainer.style.right = width + 'px';
} else {
// hide panel to the right of window
panel.style.right = width + 'px';
}
}
// LOCK/OPEN
function lockPanel() {
panel.dataset.locked = 'true';
setPanelWidth(store().width);
// anchor panel to right of window
panel.style.right = 0;
// remove handle
const handle = panel.nextElementSibling;
handle?.remove();
// reset animation styles
panel.style.opacity = '';
panel.style.transform = '';
// hover event listeners
disableHideListener();
// call panel's onLock function
if (curPanel.js?.onLock) curPanel.js.onLock();
} }
function unlockPanel(animate) { function unlockPanel(animate) {
panel.dataset.locked = 'false'; panel.dataset.locked = 'false';
setPanelWidth(store().width); setPanelWidth(store().width);
// hide panel container
container.style.width = 0;
// reset notion elements to full page
frame.style.paddingRight = 0;
notionSidebarContainer.style.right = 0;
// add handle
const handle = createElement(
'<div class="enhancer-panel--hover-handle"></div>'
)
panel.after(handle);
const addListeners = () => {
// show panel when handle is hovered
handle.addEventListener('mouseover', showPanel);
handle.addEventListener('mousemove', showPanel);
// hide panel when mouse leaves panel or handle
enableHideListener();
handle.addEventListener('mouseleave', e => {
// don't hide if mouseover scrollbar or panel
if (e.relatedTarget?.closest(".enhancer-panel--container") ||
e.relatedTarget?.classList.contains("notion-scroller")) return;
hidePanel(e);
});
}
// unlock animation
if (animate) { if (animate) {
panel.animate( panel.animate(
[ [
{ opacity: 1, transform: 'none' }, { opacity: 1, transform: 'none' },
{ opacity: 1, transform: 'translateY(60px)', offset: 0.4}, { opacity: 1, transform: 'translateY(60px)', offset: 0.4},
{ opacity: 0, transform: `translateX(${store().width - 30}px) translateY(60px)`}, { opacity: 0, transform: `translateX(${store().width}px) translateY(60px)`},
], ],
{ duration: 600, easing: 'ease-out' } { duration: 600, easing: 'ease-out' }
).onfinish = () => { ).onfinish = () => addListeners();
panel.addEventListener('mouseover', showPanel); } else addListeners();
panel.addEventListener('mouseleave', hidePanel);
}
} else {
panel.addEventListener('mouseover', showPanel);
panel.addEventListener('mouseleave', hidePanel);
}
hidePanel(); hidePanel();
if (curPanel.js && curPanel.js.onUnlock) { // call panel's onUnlock function
curPanel.js.onUnlock(); if (curPanel.js?.onUnlock) curPanel.js.onUnlock();
}
}
function lockPanel() {
panel.dataset.locked = 'true';
setPanelWidth(store().width);
// Reset animation styles
panel.style.opacity = '';
panel.style.transform = '';
// Hover event listeners
panel.removeEventListener('mouseover', showPanel);
panel.removeEventListener('mouseleave', hidePanel);
if (curPanel.js && curPanel.js.onLock) {
curPanel.js.onLock();
}
} }
function togglePanel(e) { function togglePanel(e) {
if (e) e.stopPropagation(); if (e) e.stopPropagation();
if (isLocked()) unlockPanel(true);
else lockPanel(); isLocked() ? unlockPanel(true) : lockPanel();
store().locked = panel.dataset.locked; store().locked = panel.dataset.locked;
} }
function showPanel() { function isLocked() {
if (!isLocked()) { return panel.dataset.locked === 'true';
}
// WHEN UNLOCKED
function showPanel(e) {
if (isLocked()) return;
if (e.shiftKey) {
hidePanel();
return;
}
panel.style.opacity = 1; panel.style.opacity = 1;
panel.style.transform = 'translateY(60px)'; panel.style.transform = 'translateY(60px)';
} }
}
function hidePanel() { function hidePanel(e) {
if (!isLocked()) { if (isLocked()) return;
if (e?.type === 'mousemove' && !e.shiftKey) return;
panel.style.opacity = 0; panel.style.opacity = 0;
panel.style.transform = `translateX(${store().width - 30}px) translateY(60px)`; panel.style.transform = `translateX(${store().width}px) translateY(60px)`;
}
} }
function renderSwitcherItem(mod) { // panel hides when leaving panel body
if (mod.panel) { // mousemove listeners to hide when holding shift
const item = createElement( function enableHideListener() {
`<div class="enhancer-panel--switcher-item" tabindex="0"> panel.addEventListener('mousemove', hidePanel);
<div class="enhancer-panel--icon">${mod.panel.icon}</div> panel.addEventListener('mouseleave', hidePanel);
<div class="enhancer-panel--title">${mod.panel.name || mod.name}</div>
</div>`
);
item.addEventListener('click', () => loadContent(mod));
return item;
} }
function disableHideListener() {
panel.addEventListener('mousemove', hidePanel);
panel.removeEventListener('mouseleave', hidePanel);
} }
// SWITCHER
function renderSwitcher() { function renderSwitcher() {
// switcher already rendered
if (document.querySelector('.enhancer-panel--overlay-container')) return; if (document.querySelector('.enhancer-panel--overlay-container')) return;
// Layer to close switcher // overlay to close switcher
const overlayContainer = createElement( const overlayContainer = createElement(
'<div class="enhancer-panel--overlay-container"></div>' '<div class="enhancer-panel--overlay-container"></div>'
); );
@ -246,7 +332,7 @@ module.exports = {
.querySelector('.notion-app-inner') .querySelector('.notion-app-inner')
.appendChild(overlayContainer); .appendChild(overlayContainer);
// Position switcher below header // position switcher below header
const rect = panel.querySelector('.enhancer-panel--header').getBoundingClientRect(); const rect = panel.querySelector('.enhancer-panel--header').getBoundingClientRect();
const div = createElement(` const div = createElement(`
<div style="position: fixed; top: ${rect.top}px; left: ${rect.left}px; width: ${rect.width}px; height: ${rect.height}px "> <div style="position: fixed; top: ${rect.top}px; left: ${rect.left}px; width: ${rect.width}px; height: ${rect.height}px ">
@ -254,7 +340,7 @@ module.exports = {
</div> </div>
`); `);
// Render switcher // initialize switcher
const switcher = createElement( const switcher = createElement(
'<div class="enhancer-panel--switcher"></div>' '<div class="enhancer-panel--switcher"></div>'
); );
@ -262,161 +348,173 @@ module.exports = {
switcher.append(renderSwitcherItem(mod)) switcher.append(renderSwitcherItem(mod))
); );
overlayContainer.appendChild(div);
div.firstElementChild.appendChild(switcher); div.firstElementChild.appendChild(switcher);
overlayContainer.appendChild(div);
// focus on first element
switcher.firstElementChild.focus(); switcher.firstElementChild.focus();
// Fade in // fade in
switcher.animate( switcher.animate(
[ {opacity: 0}, {opacity: 1} ], [ {opacity: 0}, {opacity: 1} ],
{ duration: 200 } { duration: 200 }
); );
// Prevent panel from closing if unlocked // prevent panel from closing if unlocked
panel.removeEventListener('mouseleave', hidePanel); disableHideListener();
// Escape key listener // keyboard shortcuts
document.addEventListener('keydown', switcherKeyEvent); document.addEventListener('keydown', switcherKeyEvent);
} }
function hideSwitcher() { function hideSwitcher() {
const overlayContainer = document.querySelector('.enhancer-panel--overlay-container'); const overlayContainer = document
.querySelector('.enhancer-panel--overlay-container');
overlayContainer.removeEventListener('click', hideSwitcher); overlayContainer.removeEventListener('click', hideSwitcher);
document.removeEventListener('keydown', switcherKeyEvent); document.removeEventListener('keydown', switcherKeyEvent);
// Fade out // fade out
document.querySelector('.enhancer-panel--switcher').animate( document.querySelector('.enhancer-panel--switcher').animate(
[ {opacity: 1}, {opacity: 0} ], [ {opacity: 1}, {opacity: 0} ],
{ duration: 200 } { duration: 200 }
).onfinish = () => overlayContainer.remove(); ).onfinish = () => overlayContainer.remove();
if (!isLocked()) panel.addEventListener('mouseleave', hidePanel); if (!isLocked()) enableHideListener();
} }
function setPanelWidth(width) { function renderSwitcherItem(mod) {
store().width = width; if (mod.panel) {
panel.style.width = width + 'px'; const item = createElement(
`<div class="enhancer-panel--switcher-item" tabindex="0">
<div class="enhancer-panel--icon">${mod.panel.icon}</div>
<div class="enhancer-panel--title">${mod.panel.name}</div>
</div>`
);
item.addEventListener('click', () => loadPanelMod(mod));
return item;
}
}
if (isLocked()) { // handle switcher hotkeys
container.style.width = width + 'px'; function switcherKeyEvent(e) {
frame.style.paddingRight = width + 'px'; e.stopPropagation();
panel.style.right = 0;
} else { // esc -> hide switcher
container.style.width = 0; if (e.key === 'Escape') return hideSwitcher();
frame.style.paddingRight = 0;
panel.style.right = width + 'px'; // space/enter -> select panel
const currentFocus = document.activeElement;
if ([' ', 'Enter'].includes(e.key)) return currentFocus.click();
// up/down/tab -> change focus
if (e.key === 'ArrowUp') focusPrevious();
else if (e.key === 'ArrowDown') focusNext();
else if (e.key === 'Tab') {
e.shiftKey ? focusPrevious() : focusNext();
e.preventDefault();
}
function focusNext() {
const nextEl = currentFocus.nextElementSibling;
(nextEl || currentFocus.parentElement.firstElementChild).focus();
}
function focusPrevious() {
const prevEl = currentFocus.previousElementSibling;
(prevEl || currentFocus.parentElement.lastElementChild).focus();
} }
} }
function enableResize() { function enableResize() {
const handle = panel.querySelector('.enhancer-panel--resize div'); const resizeHandle = panel.querySelector('.enhancer-panel--resize div');
handle.addEventListener('mousedown', initDrag); resizeHandle.addEventListener('mousedown', initDrag);
let startX, startWidth; let startX, startWidth;
const div = createElement(
const overlay = createElement(
'<div style="position: absolute; top: 0; left: 0; right: 0; bottom: 0; z-index: 99;"></div>' '<div style="position: absolute; top: 0; left: 0; right: 0; bottom: 0; z-index: 99;"></div>'
); );
// mousedown
function initDrag(e) { function initDrag(e) {
// initialize start position
startX = e.clientX; startX = e.clientX;
startWidth = store().width; startWidth = store().width;
panel.appendChild(div); panel.appendChild(overlay);
// Set transitions // set transitions
container.style.transition = 'width 50ms ease-in'; container.style.transition = 'width 50ms ease-in';
panel.style.transition = 'width 50ms ease-in, right 50ms ease-in'; panel.style.transition = 'width 50ms ease-in, right 50ms ease-in';
frame.style.transition = 'padding-right 50ms ease-in'; frame.style.transition = 'padding-right 50ms ease-in';
notionSidebarContainer.style.transition = 'padding-right 50ms ease-in';
handle.style.cursor = ''; resizeHandle.style.cursor = '';
// Prevent panel from closing if unlocked
panel.removeEventListener('mouseleave', hidePanel);
document.body.addEventListener('mousemove', drag); document.body.addEventListener('mousemove', drag);
document.body.addEventListener('mouseup', stopDrag); document.body.addEventListener('mouseup', stopDrag);
} }
// mousemove
function drag(e) { function drag(e) {
e.preventDefault(); e.preventDefault();
let width = startWidth + (startX - e.clientX); let width = startWidth + (startX - e.clientX);
// minmax
if (width < 190) width = 190; if (width < 190) width = 190;
if (width > 480) width = 480; if (width > 480) width = 480;
setPanelWidth(width); setPanelWidth(width);
if (curPanel.js && curPanel.js.onResize) { // prevent panel from closing if unlocked
curPanel.js.onResize(); disableHideListener();
}
// call panel's onResize function
if (curPanel.js?.onResize) curPanel.js.onResize();
} }
// mouseup
function stopDrag() { function stopDrag() {
handle.style.cursor = 'col-resize'; resizeHandle.style.cursor = 'col-resize';
panel.removeChild(div); panel.removeChild(overlay);
// Reset transitions // reset transitions
container.style.transition = container.style.transition =
panel.style.transition = panel.style.transition =
frame.style.transition = ''; frame.style.transition =
notionSidebarContainer.style.transition = '';
if (!isLocked()) panel.addEventListener('mouseleave', hidePanel);
if (!isLocked()) enableHideListener();
document.body.removeEventListener('mousemove', drag); document.body.removeEventListener('mousemove', drag);
document.body.removeEventListener('mouseup', stopDrag); document.body.removeEventListener('mouseup', stopDrag);
} }
} }
function isLocked() {
if (panel.dataset.locked === 'true') return true;
else return false;
}
function switcherKeyEvent(e) {
e.stopPropagation();
if (e.key === 'Escape') return hideSwitcher();
const currentFocus = document.activeElement;
if ([' ', 'Enter'].includes(e.key)) return currentFocus.click();
const focusNext = () => {
const nextEl = currentFocus.nextElementSibling;
if (nextEl) nextEl.focus();
else currentFocus.parentElement.firstElementChild.focus();
}
const focusPrevious = () => {
const prevEl = currentFocus.previousElementSibling;
if (prevEl) prevEl.focus();
else currentFocus.parentElement.lastElementChild.focus();
}
if (e.key === 'ArrowUp') focusPrevious();
else if (e.key === 'ArrowDown') focusNext();
else if (e.key === 'Tab') {
if (e.shiftKey) focusPrevious();
else focusNext();
e.preventDefault();
}
}
} }
}); });
// INITIALIZATION FUNCTIONS // INITIALIZATION FUNCTIONS
// set values for panel
async function initMod(mod) { async function initMod(mod) {
// load panel sites // load panel sites
if (mod.id === '0d541743-eb2c-4d77-83a8-3b2f5e8e5dff') { if (mod.id === '0d541743-eb2c-4d77-83a8-3b2f5e8e5dff') {
panelMods = panelMods.filter(panelMod => panelMod !== mod); panelMods = panelMods.filter(panelMod => panelMod !== mod);
return panelMods.push(...initPanelSites(mod)); return panelMods.push(...initPanelSites(mod));
} }
try { try {
if (typeof mod.panel === 'object') { if (typeof mod.panel === 'object') {
// html
// html -> relative path to html file
mod.panel.html = await fs.readFile( mod.panel.html = await fs.readFile(
path.resolve(__dirname, `../${mod.dir}/${mod.panel.html}`) path.resolve(__dirname, `../${mod.dir}/${mod.panel.html}`)
); );
// name // name
if (!mod.panel.name) mod.panel.name = mod.name; mod.panel.name = mod.panel.name || mod.name;
// icon // icon
if (mod.panel.icon) { if (mod.panel.icon) {
const iconPath = path.resolve(__dirname, `../${mod.dir}/${mod.panel.icon}`); const iconPath = path.resolve(__dirname, `../${mod.dir}/${mod.panel.icon}`);
@ -425,25 +523,33 @@ module.exports = {
} else { } else {
mod.panel.icon = mod.panel.name[0]; mod.panel.icon = mod.panel.name[0];
} }
// js
// js -> relative path to js file
if (mod.panel.js) { if (mod.panel.js) {
const jsPath = `../${mod.dir}/${mod.panel.js}`; const jsPath = `../${mod.dir}/${mod.panel.js}`;
if (await fs.pathExists(path.resolve(__dirname, jsPath))) { if (await fs.pathExists(path.resolve(__dirname, jsPath))) {
// execute js file
mod.panel.js = require(jsPath)(loadStore(mod), __exports); mod.panel.js = require(jsPath)(loadStore(mod), __exports);
} }
} }
} else if (typeof mod.panel === 'string') { } else if (typeof mod.panel === 'string') {
// icon
mod.panel.icon = mod.name[0]; mod.panel.icon = mod.name[0];
// mod.panel -> rel path to html file
mod.panel.html = await fs.readFile( mod.panel.html = await fs.readFile(
path.resolve(__dirname, `../${mod.dir}/${mod.panel}`) path.resolve(__dirname, `../${mod.dir}/${mod.panel}`)
); );
} else throw Error; } else throw Error;
} catch (err) { } catch (err) {
// remove mod from panel list
console.log('invalid panel mod: ' + mod.name); console.log('invalid panel mod: ' + mod.name);
panelMods = panelMods.filter(panelMod => panelMod !== mod); panelMods = panelMods.filter(panelMod => panelMod !== mod);
} }
} }
// returns an array of panels
function initPanelSites(mod) { function initPanelSites(mod) {
let panelSites = []; let panelSites = [];
const sitesPath = store(mod.id).sites; const sitesPath = store(mod.id).sites;
@ -451,21 +557,24 @@ module.exports = {
try { try {
const sites = require(sitesPath); const sites = require(sitesPath);
const invalid = false; const invalid = false;
// execute panel-sites/panel.js
const sitePanelJs = require('../panel-sites/panel.js')(loadStore(mod), __exports); const sitePanelJs = require('../panel-sites/panel.js')(loadStore(mod), __exports);
const frameUrl = function(url, mobile) { // returns site's iframe
if (!/(^https?:\/\/)/.test(url)) url = 'https://' + url; const frameUrl = (url, mobile) => {
if (!/(^https?:\/\/)/i.test(url)) url = 'https://' + url;
return `<iframe src=${url} class="panel-site" ${mobile ? 'mobile-user-agent' : ''}></iframe>`; return `<iframe src=${url} class="panel-site" ${mobile ? 'mobile-user-agent' : ''}></iframe>`;
} }
sites.forEach(site => { sites.forEach(site => {
if (site.url && site.name) { if (site.url && site.name) {
// get url and icon // get iframe and icon
const iframe = frameUrl(site.url, site.mobile); const iframe = frameUrl(site.url, site.mobile);
const icon = `<img style="width: 100%; height: 100%;" const icon = `<img style="width: 100%; height: 100%;"
src="${site.icon || `https://www.google.com/s2/favicons?domain=${site.url}`}" />`; src="${site.icon || `https://www.google.com/s2/favicons?domain=${site.url}`}" />`;
// add panel object to array
const panelMod = { const panelMod = {
id: `${mod.id}-${site.url}`, id: `${mod.id}-${site.url}`,
panel: { panel: {
@ -478,7 +587,7 @@ module.exports = {
}, },
} }
panelSites.push(panelMod); panelSites.push(panelMod);
} else invalid = true; } else invalid = true; // continue initializing next sites
}); });
if (invalid) throw Error; if (invalid) throw Error;
} }

View File

@ -170,13 +170,16 @@
transform: rotateZ(90deg); transform: rotateZ(90deg);
} }
.notion-collection_view-block[data-tweaks*="[toggle]"] > .notion-scroller { .notion-collection_view-block[data-tweaks*="[toggle]"] > div > .notion-scroller {
transition: height 200ms ease-in, opacity 200ms ease-in; transition: height 200ms ease-in, opacity 200ms ease-in;
} }
.notion-collection_view-block[data-tweaks*="[toggle]"][data-toggled-hidden="true"] > .notion-scroller { .notion-collection_view-block[data-tweaks*="[toggle]"][data-toggled-hidden="true"] > div > .notion-scroller {
opacity: 0; opacity: 0;
height: 0 !important; height: 0 !important;
} }
.notion-collection_view-block[data-tweaks*="[toggle]"][data-toggled-hidden="true"]
[data-hide-items] [class$="view"] > .notion-collection_view-block,
.notion-collection_view-block[data-tweaks*="[toggle]"][data-toggled-hidden="true"] .notion-collection_view-block[data-tweaks*="[toggle]"][data-toggled-hidden="true"]
[data-hide-items] .notion-collection-item { [data-hide-items] .notion-collection-item {
display: none !important; display: none !important;
@ -237,7 +240,7 @@
.notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"]) .notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"])
.notion-list-view > .notion-collection_view-block > [role="button"]:not(.notion-collection-item), .notion-list-view > .notion-collection_view-block > [role="button"]:not(.notion-collection-item),
.notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"]) .notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"])
.notion-gallery-view > .notion-collection_view-block > [role="button"]:not(.notion-collection-item) { .notion-gallery-view > .notion-collection_view-block [style*="grid"] > [role="button"] {
display: none !important; display: none !important;
} }
.notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"]) .notion-collection_view-block[data-tweaks]:not([data-tweaks*="[new_item]"])

View File

@ -14,7 +14,7 @@ module.exports = {
tags: ['extension'], tags: ['extension'],
name: 'simpler databases', name: 'simpler databases',
desc: 'adds a menu to inline databases to toggle ui elements.', desc: 'adds a menu to inline databases to toggle ui elements.',
version: '1.0.0', version: '1.0.2',
author: 'CloudHill', author: 'CloudHill',
hacks: { hacks: {
'renderer/preload.js'(store, __exports) { 'renderer/preload.js'(store, __exports) {
@ -157,7 +157,7 @@ module.exports = {
for (let { addedNodes } of list) { for (let { addedNodes } of list) {
if ( if (
addedNodes[0] && addedNodes[0] &&
addedNodes[0].className === 'notion-page-content' addedNodes[0].className === 'notion-presence-container'
) { ) {
findInlineCollections(); findInlineCollections();
contentObserver.disconnect(); contentObserver.disconnect();