tabs: inter-window drag, titles & icons

This commit is contained in:
dragonwocky 2021-12-10 23:36:56 +11:00
parent 2dcfef0b6b
commit cb7838350f
9 changed files with 177 additions and 54 deletions

View File

@ -1,6 +1,6 @@
/** /**
* notion-enhancer: font chooser * notion-enhancer: font chooser
* (c) 2021 TorchAtlas (https://bit.ly/torchatlas/) * (c) 2021 TorchAtlas (https://github.com/torchatlas/)
* (c) 2021 admiraldus (https://github.com/admiraldus) * (c) 2021 admiraldus (https://github.com/admiraldus)
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/) * (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license * (https://notion-enhancer.github.io/) under the MIT license

View File

@ -1,6 +1,6 @@
/** /**
* notion-enhancer: font chooser * notion-enhancer: font chooser
* (c) 2021 TorchAtlas (https://bit.ly/torchatlas/) * (c) 2021 TorchAtlas (https://github.com/torchatlas/)
* (c) 2021 admiraldus (https://github.com/admiraldus * (c) 2021 admiraldus (https://github.com/admiraldus
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/) * (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license * (https://notion-enhancer.github.io/) under the MIT license

View File

@ -7,7 +7,7 @@
"authors": [ "authors": [
{ {
"name": "TorchAtlas", "name": "TorchAtlas",
"homepage": "https://bit.ly/torchatlas/", "homepage": "https://github.com/torchatlas/",
"avatar": "https://avatars.githubusercontent.com/u/12666855" "avatar": "https://avatars.githubusercontent.com/u/12666855"
} }
], ],

View File

@ -15,14 +15,6 @@ import { createWindowButtons } from './buttons.mjs';
windowActionsSelector = '#window-actions'; windowActionsSelector = '#window-actions';
await web.whenReady([windowActionsSelector]); await web.whenReady([windowActionsSelector]);
// const $tabs = document.querySelector(topbarActionsSelector),
// $dragarea = web.html`<div class="integrated_titlebar--dragarea"></div>`;
// $tabs.prepend($dragarea);
// document.documentElement.style.setProperty(
// '--integrated_titlebar--dragarea-height',
// dragareaHeight + 'px'
// );
const $topbarActions = document.querySelector(windowActionsSelector), const $topbarActions = document.querySelector(windowActionsSelector),
$windowButtons = await createWindowButtons(api, db); $windowButtons = await createWindowButtons(api, db);
web.render($topbarActions, $windowButtons); web.render($topbarActions, $windowButtons);

40
repo/tabs/client.mjs Normal file
View File

@ -0,0 +1,40 @@
/**
* notion-enhancer: theming
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
'use strict';
export default async function ({ electron }, db) {
let title = '',
icon = '';
const notionSetWindowTitle = __electronApi.setWindowTitle,
imgIconSelector =
'.notion-topbar > div > :nth-child(2) > .notion-focusable:last-child .notion-record-icon img',
nativeIconSelector =
'.notion-topbar > div > :nth-child(2) > .notion-focusable:last-child .notion-record-icon [role="image"]',
getIcon = () => {
const $imgIcon = document.querySelector(imgIconSelector),
$nativeIcon = document.querySelector(nativeIconSelector);
if ($imgIcon) {
return $imgIcon.style.background.replace(
/url\("\/images/,
'url("notion://www.notion.so/images'
);
}
if ($nativeIcon) return $nativeIcon.ariaLabel;
return '';
},
updateTitle = (newTitle = title) => {
if (!newTitle) return;
title = newTitle;
icon = getIcon();
electron.sendMessageToHost('set-tab-title', title);
electron.sendMessageToHost('set-tab-icon', icon);
notionSetWindowTitle(title);
};
__electronApi.setWindowTitle = (newTitle) => updateTitle(newTitle);
document.addEventListener('focus', updateTitle);
electron.onMessage('trigger-title-update', () => updateTitle());
}

14
repo/tabs/main.cjs Normal file
View File

@ -0,0 +1,14 @@
/**
* notion-enhancer: tabs
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
'use strict';
module.exports = async function ({}, db, __exports, __eval) {
const electron = require('electron');
electron.ipcMain.on('notion-enhancer:close-tab', (event, { window, id }) => {
electron.webContents.fromId(window).send('notion-enhancer:close-tab', id);
});
};

View File

@ -17,7 +17,11 @@
"frame": ["tabs.css"] "frame": ["tabs.css"]
}, },
"js": { "js": {
"electron": [{ "source": "rendererIndex.cjs", "target": "renderer/index.js" }] "client": ["client.mjs"],
"electron": [
{ "source": "main.cjs", "target": "main/main.js" },
{ "source": "rendererIndex.cjs", "target": "renderer/index.js" }
]
}, },
"options": [ "options": [
{ {

View File

@ -15,6 +15,8 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
let focusedTab, xIcon; let focusedTab, xIcon;
const tabCache = new Map(); const tabCache = new Map();
class Tab { class Tab {
id = fmt.uuidv4();
$notion = web.html` $notion = web.html`
<webview class="notion-webview" partition="persist:notion" <webview class="notion-webview" partition="persist:notion"
preload="file://${fs.notionPath('renderer/preload.js')}" preload="file://${fs.notionPath('renderer/preload.js')}"
@ -29,16 +31,29 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
></webview> ></webview>
`; `;
$tabTitle = web.html`<span class="tab-title">v0.11.0 plan v0.11.0 plan v0.11.0 plan v0.11.0 plan</span>`; $tabIcon = web.html`<span class="tab-icon"></span>`;
$tabTitle = web.html`<span class="tab-title"></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`<div class="tab" draggable="true" id="${fmt.uuidv4()}"></div>`, web.html`<div class="tab" draggable="true" id="${this.id}"></div>`,
this.$tabIcon,
this.$tabTitle, this.$tabTitle,
this.$closeTab this.$closeTab
); );
constructor($tabs, $root, notionUrl = 'notion://www.notion.so/') { constructor(
$tabs,
$root,
{
notionUrl = 'notion://www.notion.so/',
cancelAnimation = false,
icon = '',
title = 'notion.so',
} = {}
) {
this.$notion.src = notionUrl; this.$notion.src = notionUrl;
this.$tabTitle.innerText = title;
this.setIcon(icon);
tabCache.set(this.$tab.id, this); tabCache.set(this.$tab.id, this);
web.render($tabs, this.$tab); web.render($tabs, this.$tab);
@ -50,21 +65,23 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
this.$tab.addEventListener('click', (event) => { this.$tab.addEventListener('click', (event) => {
if (event.target !== this.$closeTab && !this.$closeTab.contains(event.target)) { if (event.target !== this.$closeTab && !this.$closeTab.contains(event.target)) {
this.focusTab(); this.focus();
} }
}); });
this.$closeTab.addEventListener('click', () => this.closeTab()); this.$closeTab.addEventListener('click', () => this.close());
this.focusTab(); if (!cancelAnimation) {
this.$tab.animate([{ width: '0px' }, { width: `${this.$tab.clientWidth}px` }], { this.$tab.animate([{ width: '0px' }, { width: `${this.$tab.clientWidth}px` }], {
duration: 100, duration: 100,
easing: 'ease-in', easing: 'ease-in',
}).finished; }).finished;
this.listenToNotion(); }
this.focus();
this.addNotionListeners();
return this; return this;
} }
async focusTab() { async focus() {
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 = '';
}); });
@ -77,9 +94,10 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
this.focusNotion(); this.focusNotion();
focusedTab = this; focusedTab = this;
} }
async closeTab() { async close() {
const $sibling = this.$tab.nextElementSibling || this.$tab.previousElementSibling; const $sibling = this.$tab.nextElementSibling || this.$tab.previousElementSibling;
if ($sibling) { if ($sibling) {
if (!focusedTab || focusedTab === this) $sibling.click();
const width = `${this.$tab.clientWidth}px`; const width = `${this.$tab.clientWidth}px`;
this.$tab.style.width = 0; this.$tab.style.width = 0;
this.$tab.style.pointerEvents = 'none'; this.$tab.style.pointerEvents = 'none';
@ -90,10 +108,21 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
this.$tab.remove(); this.$tab.remove();
this.$notion.remove(); this.$notion.remove();
this.$search.remove(); this.$search.remove();
if (focusedTab === this) $sibling.click();
} else electronWindow.close(); } else electronWindow.close();
} }
setIcon(icon) {
if (icon.startsWith('url(')) {
// img
this.$tabIcon.style.background = icon;
this.$tabIcon.innerText = '';
} else {
// unicode (native)
this.$tabIcon.innerText = icon;
this.$tabIcon.style.background = '';
}
}
webContents() { webContents() {
return electron.remote.webContents.fromId(this.$notion.getWebContentsId()); return electron.remote.webContents.fromId(this.$notion.getWebContentsId());
} }
@ -101,6 +130,9 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
document.activeElement?.blur?.(); document.activeElement?.blur?.();
this.$notion.blur(); this.$notion.blur();
this.$notion.focus(); this.$notion.focus();
requestAnimationFrame(() => {
notionIpc.sendIndexToNotion(this.$notion, 'notion-enhancer:trigger-title-update');
});
} }
focusSearch() { focusSearch() {
document.activeElement?.blur?.(); document.activeElement?.blur?.();
@ -108,7 +140,7 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
this.$search.focus(); this.$search.focus();
} }
listenToNotion() { addNotionListeners() {
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) =>
@ -149,6 +181,11 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
fromNotion('zoom', (zoomFactor) => { fromNotion('zoom', (zoomFactor) => {
this.webContents().setZoomFactor(zoomFactor); this.webContents().setZoomFactor(zoomFactor);
}); });
fromNotion('notion-enhancer:set-tab-title', (title) => {
this.$tabTitle.innerText = title;
});
fromNotion('notion-enhancer:set-tab-icon', (icon) => this.setIcon(icon));
} }
#firstQuery = true; #firstQuery = true;
@ -194,65 +231,90 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, __export
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');
$newTab.addEventListener('click', () => {
new Tab($tabs, $root);
});
electron.ipcRenderer.on('notion-enhancer:close-tab', (event, id) => {
const tab = tabCache.get(id);
if (tab) tab.close();
});
new Tab($tabs, $root, {
notionUrl: url.parse(window.location.href, true).query.path,
cancelAnimation: true,
});
let $draggedTab; let $draggedTab;
const getDragTarget = ($el) => { const getDragTarget = ($el) => {
while (!$el.matches('.tab, header, body')) $el = $el.parentElement; while (!$el.matches('.tab, header, body')) $el = $el.parentElement;
if ($el.matches('header')) $el = $el.firstElementChild; if ($el.matches('header')) $el = $el.firstElementChild;
return $el.matches('#tabs, .tab') ? $el : undefined; return $el.matches('#tabs, .tab') ? $el : undefined;
}, },
resetTabs = () => { clearDragStatus = () => {
document document
.querySelectorAll('.dragged-over') .querySelectorAll('.dragged-over')
.forEach(($el) => $el.classList.remove('dragged-over')); .forEach(($el) => $el.classList.remove('dragged-over'));
},
resetDraggedTabs = () => {
if ($draggedTab) {
clearDragStatus();
$draggedTab.style.opacity = '';
$draggedTab = undefined;
}
}; };
$header.addEventListener('dragstart', (event) => { $header.addEventListener('dragstart', (event) => {
$draggedTab = getDragTarget(event.target); $draggedTab = getDragTarget(event.target);
$draggedTab.style.opacity = 0.5; $draggedTab.style.opacity = 0.5;
const tab = tabCache.get($draggedTab.id);
event.dataTransfer.setData( event.dataTransfer.setData(
'text', 'text',
JSON.stringify({ JSON.stringify({
window: electronWindow.webContents.id, window: electronWindow.webContents.id,
tab: $draggedTab.id, tab: $draggedTab.id,
title: $draggedTab.children[0].innerText, icon: tab.$tabIcon.innerText || tab.$tabIcon.style.background,
url: tabCache.get($draggedTab.id).$notion.src, title: tab.$tabTitle.innerText,
url: tab.$notion.src,
}) })
); );
}); });
$header.addEventListener('dragover', (event) => { $header.addEventListener('dragover', (event) => {
const $target = getDragTarget(event.target); const $target = getDragTarget(event.target);
if ($target) { if ($target) {
resetTabs(); clearDragStatus();
$target.classList.add('dragged-over'); $target.classList.add('dragged-over');
event.preventDefault(); event.preventDefault();
} }
}); });
$header.addEventListener('dragend', (event) => {
resetTabs();
$draggedTab.style.opacity = '';
$draggedTab = undefined;
});
document.addEventListener('drop', (event) => { document.addEventListener('drop', (event) => {
const eventData = JSON.parse(event.dataTransfer.getData('text')), const eventData = JSON.parse(event.dataTransfer.getData('text')),
$target = getDragTarget(event.target) || $tabs,
sameWindow = eventData.window === electronWindow.webContents.id, sameWindow = eventData.window === electronWindow.webContents.id,
$target = getDragTarget(event.target), tabMovement =
movement = !sameWindow ||
$target && ($target &&
(!sameWindow || $target !== $draggedTab &&
($target !== $draggedTab && $target !== $draggedTab.nextElementSibling &&
$target !== $draggedTab.nextElementSibling && ($target.matches('#tabs') ? $target.lastElementChild !== $draggedTab : true));
($target.matches('#tabs') ? $target.lastElementChild !== $draggedTab : true))); if (!sameWindow) {
if (movement) { electron.ipcRenderer.send('notion-enhancer:close-tab', {
if (sameWindow) { window: eventData.window,
if ($target.matches('#tabs')) { id: eventData.tab,
$target.append($draggedTab); });
} else $target.before($draggedTab); const transferred = new Tab($tabs, $root, {
} notionUrl: eventData.url,
cancelAnimation: true,
icon: eventData.icon,
title: eventData.title,
});
$draggedTab = transferred.$tab;
} }
if (tabMovement) {
if ($target.matches('#tabs')) {
$target.append($draggedTab);
} else $target.before($draggedTab);
}
resetDraggedTabs();
}); });
$header.addEventListener('dragend', (event) => resetDraggedTabs());
$newTab.addEventListener('click', () => {
new Tab($tabs, $root, url.parse(window.location.href, true).query.path);
});
$newTab.click();
}; };
}; };

View File

@ -42,6 +42,7 @@ header {
display: flex; display: flex;
flex-grow: 1; flex-grow: 1;
flex-shrink: 1; flex-shrink: 1;
width: 14em;
max-width: 14em; max-width: 14em;
overflow: hidden; overflow: hidden;
padding: 6.4px 9.6px 6.4px 9.6px; padding: 6.4px 9.6px 6.4px 9.6px;
@ -55,6 +56,16 @@ 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;
} }
.tab .tab-icon {
font-size: 14px;
margin-right: 6px;
}
.tab .tab-icon[style*='background'] {
width: 14px;
height: 14px;
align-self: center;
margin-right: 8px;
}
.tab .tab-title { .tab .tab-title {
white-space: nowrap; white-space: nowrap;
overflow: hidden; overflow: hidden;