mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-06 05:29:02 +00:00
draggable & slide-in/out tabs
This commit is contained in:
parent
58b44d556e
commit
2dcfef0b6b
@ -234,8 +234,8 @@ import './styles.mjs';
|
|||||||
const mode = mod.tags.includes('light') ? 'light' : 'dark',
|
const mode = mod.tags.includes('light') ? 'light' : 'dark',
|
||||||
id = mod.id,
|
id = mod.id,
|
||||||
mods = await registry.list(
|
mods = await registry.list(
|
||||||
(mod) =>
|
async (mod) =>
|
||||||
mod.environments.includes(env.name) &&
|
(await registry.enabled(mod.id)) &&
|
||||||
mod.tags.includes('theme') &&
|
mod.tags.includes('theme') &&
|
||||||
mod.tags.includes(mode) &&
|
mod.tags.includes(mode) &&
|
||||||
mod.id !== id
|
mod.id !== id
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
module.exports = async function ({ components, env, web, fs }, db, __exports, __eval) {
|
module.exports = async function ({ components, env, web, fmt, fs }, db, __exports, __eval) {
|
||||||
const url = require('url'),
|
const url = require('url'),
|
||||||
electron = require('electron'),
|
electron = require('electron'),
|
||||||
electronWindow = electron.remote.getCurrentWindow(),
|
electronWindow = electron.remote.getCurrentWindow(),
|
||||||
@ -32,14 +32,14 @@ module.exports = async function ({ components, env, web, fs }, db, __exports, __
|
|||||||
$tabTitle = web.html`<span class="tab-title">v0.11.0 plan v0.11.0 plan v0.11.0 plan v0.11.0 plan</span>`;
|
$tabTitle = web.html`<span class="tab-title">v0.11.0 plan v0.11.0 plan v0.11.0 plan v0.11.0 plan</span>`;
|
||||||
$closeTab = web.html`<span class="tab-close">${xIcon}</span>`;
|
$closeTab = web.html`<span class="tab-close">${xIcon}</span>`;
|
||||||
$tab = web.render(
|
$tab = web.render(
|
||||||
web.html`<button class="tab" draggable="true"></button>`,
|
web.html`<div class="tab" draggable="true" id="${fmt.uuidv4()}"></div>`,
|
||||||
this.$tabTitle,
|
this.$tabTitle,
|
||||||
this.$closeTab
|
this.$closeTab
|
||||||
);
|
);
|
||||||
|
|
||||||
constructor($tabs, $root, notionUrl = 'notion://www.notion.so/') {
|
constructor($tabs, $root, notionUrl = 'notion://www.notion.so/') {
|
||||||
this.$notion.src = notionUrl;
|
this.$notion.src = notionUrl;
|
||||||
tabCache.set($tab, this);
|
tabCache.set(this.$tab.id, this);
|
||||||
|
|
||||||
web.render($tabs, this.$tab);
|
web.render($tabs, this.$tab);
|
||||||
web.render($root, this.$search);
|
web.render($root, this.$search);
|
||||||
@ -56,11 +56,15 @@ module.exports = async function ({ components, env, web, fs }, db, __exports, __
|
|||||||
this.$closeTab.addEventListener('click', () => this.closeTab());
|
this.$closeTab.addEventListener('click', () => this.closeTab());
|
||||||
|
|
||||||
this.focusTab();
|
this.focusTab();
|
||||||
this.listen();
|
this.$tab.animate([{ width: '0px' }, { width: `${this.$tab.clientWidth}px` }], {
|
||||||
|
duration: 100,
|
||||||
|
easing: 'ease-in',
|
||||||
|
}).finished;
|
||||||
|
this.listenToNotion();
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
focusTab() {
|
async focusTab() {
|
||||||
document.querySelectorAll('.notion-webview, .search-webview').forEach(($webview) => {
|
document.querySelectorAll('.notion-webview, .search-webview').forEach(($webview) => {
|
||||||
if (![this.$notion, this.$search].includes($webview)) $webview.style.display = '';
|
if (![this.$notion, this.$search].includes($webview)) $webview.style.display = '';
|
||||||
});
|
});
|
||||||
@ -73,14 +77,21 @@ module.exports = async function ({ components, env, web, fs }, db, __exports, __
|
|||||||
this.focusNotion();
|
this.focusNotion();
|
||||||
focusedTab = this;
|
focusedTab = this;
|
||||||
}
|
}
|
||||||
closeTab() {
|
async closeTab() {
|
||||||
const $sibling = this.$tab.nextElementSibling || this.$tab.previousElementSibling;
|
const $sibling = this.$tab.nextElementSibling || this.$tab.previousElementSibling;
|
||||||
if ($sibling) {
|
if ($sibling) {
|
||||||
|
const width = `${this.$tab.clientWidth}px`;
|
||||||
|
this.$tab.style.width = 0;
|
||||||
|
this.$tab.style.pointerEvents = 'none';
|
||||||
|
await this.$tab.animate([{ width }, { width: '0px' }], {
|
||||||
|
duration: 100,
|
||||||
|
easing: 'ease-out',
|
||||||
|
}).finished;
|
||||||
this.$tab.remove();
|
this.$tab.remove();
|
||||||
this.$notion.remove();
|
this.$notion.remove();
|
||||||
this.$search.remove();
|
this.$search.remove();
|
||||||
if (focusedTab === this) $sibling.click();
|
if (focusedTab === this) $sibling.click();
|
||||||
}
|
} else electronWindow.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
webContents() {
|
webContents() {
|
||||||
@ -97,7 +108,7 @@ module.exports = async function ({ components, env, web, fs }, db, __exports, __
|
|||||||
this.$search.focus();
|
this.$search.focus();
|
||||||
}
|
}
|
||||||
|
|
||||||
listen() {
|
listenToNotion() {
|
||||||
const fromNotion = (channel, listener) =>
|
const fromNotion = (channel, listener) =>
|
||||||
notionIpc.receiveIndexFromNotion.addListener(this.$notion, channel, listener),
|
notionIpc.receiveIndexFromNotion.addListener(this.$notion, channel, listener),
|
||||||
fromSearch = (channel, listener) =>
|
fromSearch = (channel, listener) =>
|
||||||
@ -177,12 +188,68 @@ module.exports = async function ({ components, env, web, fs }, db, __exports, __
|
|||||||
window['__start'] = async () => {
|
window['__start'] = async () => {
|
||||||
const $header = web.html`<header></header>`,
|
const $header = web.html`<header></header>`,
|
||||||
$tabs = web.html`<div id="tabs"></div>`,
|
$tabs = web.html`<div id="tabs"></div>`,
|
||||||
$newTab = web.html`<button class="new-tab">${await components.feather('plus')}</button>`,
|
$newTab = web.html`<div class="new-tab">${await components.feather('plus')}</div>`,
|
||||||
$root = document.querySelector('#root'),
|
$root = document.querySelector('#root'),
|
||||||
$windowActions = web.html`<div id="window-actions"></div>`;
|
$windowActions = web.html`<div id="window-actions"></div>`;
|
||||||
document.body.prepend(web.render($header, $tabs, $newTab, $windowActions));
|
document.body.prepend(web.render($header, $tabs, $newTab, $windowActions));
|
||||||
xIcon = await components.feather('x');
|
xIcon = await components.feather('x');
|
||||||
|
|
||||||
|
let $draggedTab;
|
||||||
|
const getDragTarget = ($el) => {
|
||||||
|
while (!$el.matches('.tab, header, body')) $el = $el.parentElement;
|
||||||
|
if ($el.matches('header')) $el = $el.firstElementChild;
|
||||||
|
return $el.matches('#tabs, .tab') ? $el : undefined;
|
||||||
|
},
|
||||||
|
resetTabs = () => {
|
||||||
|
document
|
||||||
|
.querySelectorAll('.dragged-over')
|
||||||
|
.forEach(($el) => $el.classList.remove('dragged-over'));
|
||||||
|
};
|
||||||
|
$header.addEventListener('dragstart', (event) => {
|
||||||
|
$draggedTab = getDragTarget(event.target);
|
||||||
|
$draggedTab.style.opacity = 0.5;
|
||||||
|
event.dataTransfer.setData(
|
||||||
|
'text',
|
||||||
|
JSON.stringify({
|
||||||
|
window: electronWindow.webContents.id,
|
||||||
|
tab: $draggedTab.id,
|
||||||
|
title: $draggedTab.children[0].innerText,
|
||||||
|
url: tabCache.get($draggedTab.id).$notion.src,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
$header.addEventListener('dragover', (event) => {
|
||||||
|
const $target = getDragTarget(event.target);
|
||||||
|
if ($target) {
|
||||||
|
resetTabs();
|
||||||
|
$target.classList.add('dragged-over');
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$header.addEventListener('dragend', (event) => {
|
||||||
|
resetTabs();
|
||||||
|
$draggedTab.style.opacity = '';
|
||||||
|
$draggedTab = undefined;
|
||||||
|
});
|
||||||
|
document.addEventListener('drop', (event) => {
|
||||||
|
const eventData = JSON.parse(event.dataTransfer.getData('text')),
|
||||||
|
sameWindow = eventData.window === electronWindow.webContents.id,
|
||||||
|
$target = getDragTarget(event.target),
|
||||||
|
movement =
|
||||||
|
$target &&
|
||||||
|
(!sameWindow ||
|
||||||
|
($target !== $draggedTab &&
|
||||||
|
$target !== $draggedTab.nextElementSibling &&
|
||||||
|
($target.matches('#tabs') ? $target.lastElementChild !== $draggedTab : true)));
|
||||||
|
if (movement) {
|
||||||
|
if (sameWindow) {
|
||||||
|
if ($target.matches('#tabs')) {
|
||||||
|
$target.append($draggedTab);
|
||||||
|
} else $target.before($draggedTab);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
$newTab.addEventListener('click', () => {
|
$newTab.addEventListener('click', () => {
|
||||||
new Tab($tabs, $root, url.parse(window.location.href, true).query.path);
|
new Tab($tabs, $root, url.parse(window.location.href, true).query.path);
|
||||||
});
|
});
|
||||||
|
@ -27,7 +27,7 @@ header {
|
|||||||
display: flex;
|
display: flex;
|
||||||
background: var(--theme--bg_secondary);
|
background: var(--theme--bg_secondary);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 12px 8px 0 8px;
|
padding: 8px;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
-webkit-app-region: drag;
|
-webkit-app-region: drag;
|
||||||
z-index: 3;
|
z-index: 3;
|
||||||
@ -35,13 +35,16 @@ header {
|
|||||||
|
|
||||||
#tabs {
|
#tabs {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
margin-bottom: -8px;
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
#tabs .tab {
|
.tab {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
flex-shrink: 1;
|
||||||
max-width: 14em;
|
max-width: 14em;
|
||||||
padding: 10.4px 9.6px 6.4px 9.6px;
|
overflow: hidden;
|
||||||
margin-top: -4px;
|
padding: 6.4px 9.6px 6.4px 9.6px;
|
||||||
|
|
||||||
color: var(--theme--text_secondary);
|
color: var(--theme--text_secondary);
|
||||||
background: var(--theme--bg);
|
background: var(--theme--bg);
|
||||||
@ -52,11 +55,27 @@ header {
|
|||||||
border-bottom: 3px solid var(--theme--ui_divider);
|
border-bottom: 3px solid var(--theme--ui_divider);
|
||||||
-webkit-app-region: no-drag;
|
-webkit-app-region: no-drag;
|
||||||
}
|
}
|
||||||
#tabs .tab .tab-title {
|
.tab .tab-title {
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
#tabs .tab .tab-close {
|
|
||||||
|
.tab:hover {
|
||||||
|
background: var(--theme--ui_interactive-hover);
|
||||||
|
}
|
||||||
|
.tab.current {
|
||||||
|
background: var(--theme--ui_interactive-active);
|
||||||
|
}
|
||||||
|
|
||||||
|
#tabs.dragged-over {
|
||||||
|
box-shadow: 2px 0 0 0 var(--theme--accent_blue-selection);
|
||||||
|
}
|
||||||
|
.tab.dragged-over {
|
||||||
|
box-shadow: inset 2px 0 0 0 var(--theme--accent_blue-selection);
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-tab,
|
||||||
|
.tab-close {
|
||||||
transition: background 20ms ease-in 0s;
|
transition: background 20ms ease-in 0s;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
@ -68,57 +87,35 @@ header {
|
|||||||
width: 20px;
|
width: 20px;
|
||||||
padding: 0 0.25px 0 0;
|
padding: 0 0.25px 0 0;
|
||||||
|
|
||||||
margin-left: auto;
|
|
||||||
border: none;
|
border: none;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
font-size: 18px;
|
font-size: 14px;
|
||||||
|
-webkit-app-region: no-drag;
|
||||||
}
|
}
|
||||||
#tabs .tab .tab-close svg {
|
.new-tab svg,
|
||||||
|
.tab-close svg {
|
||||||
width: 14px;
|
width: 14px;
|
||||||
height: 14px;
|
height: 14px;
|
||||||
fill: var(--theme--icon_secondary);
|
fill: var(--theme--icon_secondary);
|
||||||
|
color: var(--theme--icon_secondary);
|
||||||
}
|
}
|
||||||
#tabs .tab:hover {
|
.new-tab:focus,
|
||||||
|
.new-tab:hover,
|
||||||
|
.tab-close:focus,
|
||||||
|
.tab-close:hover {
|
||||||
background: var(--theme--ui_interactive-hover);
|
background: var(--theme--ui_interactive-hover);
|
||||||
}
|
}
|
||||||
#tabs .tab.current {
|
.new-tab:active,
|
||||||
|
.tab-close:active {
|
||||||
background: var(--theme--ui_interactive-active);
|
background: var(--theme--ui_interactive-active);
|
||||||
}
|
}
|
||||||
|
|
||||||
.new-tab {
|
.new-tab {
|
||||||
transition: background 20ms ease-in 0s;
|
align-self: center;
|
||||||
cursor: pointer;
|
margin: 0 48px -4px 6px;
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
flex-shrink: 0;
|
|
||||||
border-radius: 3px;
|
|
||||||
height: 28px;
|
|
||||||
width: 33px;
|
|
||||||
padding: 0 0.25px 0 0;
|
|
||||||
|
|
||||||
margin-left: 6px;
|
|
||||||
border: none;
|
|
||||||
background: transparent;
|
|
||||||
font-size: 18px;
|
|
||||||
-webkit-app-region: no-drag;
|
|
||||||
}
|
}
|
||||||
.new-tab svg {
|
.tab-close {
|
||||||
display: block;
|
margin-left: auto;
|
||||||
width: 20px;
|
|
||||||
height: 20px;
|
|
||||||
fill: var(--theme--icon);
|
|
||||||
color: var(--theme--icon);
|
|
||||||
}
|
|
||||||
.new-tab:focus,
|
|
||||||
.new-tab:hover,
|
|
||||||
#tabs .tab-close:focus,
|
|
||||||
#tabs .tab-close:hover {
|
|
||||||
background: var(--theme--ui_interactive-hover);
|
|
||||||
}
|
|
||||||
.new-tab:active,
|
|
||||||
#tabs .tab-close:active {
|
|
||||||
background: var(--theme--ui_interactive-active);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#window-actions {
|
#window-actions {
|
||||||
@ -131,13 +128,11 @@ header {
|
|||||||
#root {
|
#root {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notion-webview {
|
.notion-webview {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-webview {
|
.search-webview {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 60px;
|
height: 60px;
|
||||||
|
@ -17,14 +17,15 @@ export default async function ({ web, registry, storage, electron }, db) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const updateTheme = async () => {
|
const updateTheme = async () => {
|
||||||
await storage.set(
|
const isDark =
|
||||||
['theme'],
|
document.querySelector('.notion-dark-theme') ||
|
||||||
document.querySelector('.notion-dark-theme') ? 'dark' : 'light'
|
document.querySelector('.notion-body.dark'),
|
||||||
);
|
isLight = document.querySelector('.notion-light-theme'),
|
||||||
const mode = await storage.get(['theme'], 'light'),
|
mode = isDark ? 'dark' : isLight ? 'light' : '';
|
||||||
inactive = mode === 'light' ? 'dark' : 'light';
|
if (!mode) return;
|
||||||
|
await storage.set(['theme'], mode);
|
||||||
document.documentElement.classList.add(mode);
|
document.documentElement.classList.add(mode);
|
||||||
document.documentElement.classList.remove(inactive);
|
document.documentElement.classList.remove(mode === 'light' ? 'dark' : 'light');
|
||||||
electron.sendMessage('update-theme');
|
electron.sendMessage('update-theme');
|
||||||
const searchThemeVars = [
|
const searchThemeVars = [
|
||||||
'bg',
|
'bg',
|
||||||
@ -47,11 +48,10 @@ export default async function ({ web, registry, storage, electron }, db) {
|
|||||||
electron.sendMessage('set-search-theme', searchThemeVars);
|
electron.sendMessage('set-search-theme', searchThemeVars);
|
||||||
};
|
};
|
||||||
web.addDocumentObserver((mutation) => {
|
web.addDocumentObserver((mutation) => {
|
||||||
const potentialThemeChange = [document.body, document.documentElement].includes(
|
const potentialThemeChange = mutation.target.matches?.('html, body, .notion-app-inner');
|
||||||
mutation.target
|
|
||||||
);
|
|
||||||
if (potentialThemeChange && document.hasFocus()) updateTheme();
|
if (potentialThemeChange && document.hasFocus()) updateTheme();
|
||||||
});
|
});
|
||||||
updateTheme();
|
updateTheme();
|
||||||
document.addEventListener('visibilitychange', updateTheme);
|
document.addEventListener('visibilitychange', updateTheme);
|
||||||
|
document.addEventListener('focus', updateTheme);
|
||||||
}
|
}
|
||||||
|
@ -10,10 +10,9 @@ export default async function ({ web, registry, storage, electron }, db) {
|
|||||||
await web.whenReady();
|
await web.whenReady();
|
||||||
|
|
||||||
const updateTheme = async () => {
|
const updateTheme = async () => {
|
||||||
const mode = await storage.get(['theme'], 'light'),
|
const mode = await storage.get(['theme'], 'light');
|
||||||
inactive = mode === 'light' ? 'dark' : 'light';
|
|
||||||
document.documentElement.classList.add(mode);
|
document.documentElement.classList.add(mode);
|
||||||
document.documentElement.classList.remove(inactive);
|
document.documentElement.classList.remove(mode === 'light' ? 'dark' : 'light');
|
||||||
};
|
};
|
||||||
document.addEventListener('visibilitychange', updateTheme);
|
document.addEventListener('visibilitychange', updateTheme);
|
||||||
electron.onMessage('update-theme', updateTheme);
|
electron.onMessage('update-theme', updateTheme);
|
||||||
|
@ -10,10 +10,9 @@ module.exports = async function ({ registry, web, storage, electron }, db, __exp
|
|||||||
await web.whenReady();
|
await web.whenReady();
|
||||||
|
|
||||||
const updateTheme = async () => {
|
const updateTheme = async () => {
|
||||||
const mode = await storage.get(['theme'], 'light'),
|
const mode = await storage.get(['theme'], 'light');
|
||||||
inactive = mode === 'light' ? 'dark' : 'light';
|
|
||||||
document.documentElement.classList.add(mode);
|
document.documentElement.classList.add(mode);
|
||||||
document.documentElement.classList.remove(inactive);
|
document.documentElement.classList.remove(mode === 'light' ? 'dark' : 'light');
|
||||||
};
|
};
|
||||||
electron.onMessage('update-theme', updateTheme);
|
electron.onMessage('update-theme', updateTheme);
|
||||||
updateTheme();
|
updateTheme();
|
||||||
|
Loading…
Reference in New Issue
Block a user