draggable & slide-in/out tabs

This commit is contained in:
dragonwocky 2021-12-10 14:40:47 +11:00
parent 58b44d556e
commit 2dcfef0b6b
6 changed files with 132 additions and 72 deletions

View File

@ -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

View File

@ -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);
}); });

View File

@ -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;

View File

@ -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);
} }

View File

@ -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);

View File

@ -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();