diff --git a/repo/tabs/client.mjs b/repo/tabs/client.mjs
index 934ab44..26aae87 100644
--- a/repo/tabs/client.mjs
+++ b/repo/tabs/client.mjs
@@ -7,6 +7,17 @@
'use strict';
export default async function ({ web, electron }, db) {
+ const newTabHotkey = await db.get(['new_tab']),
+ closeTabHotkey = await db.get(['close_tab']),
+ selectTabModifier = await db.get(['select_modifier']);
+ web.addHotkeyListener(newTabHotkey, () => electron.sendMessageToHost('new-tab'));
+ web.addHotkeyListener(closeTabHotkey, () => electron.sendMessageToHost('close-tab'));
+ for (let i = 1; i < 10; i++) {
+ web.addHotkeyListener([selectTabModifier, i.toString()], () => {
+ electron.sendMessageToHost('select-tab', i);
+ });
+ }
+
const breadcrumbSelector =
'.notion-topbar > div > :nth-child(2) > .notion-focusable:last-child',
imgIconSelector = `${breadcrumbSelector} .notion-record-icon img`,
diff --git a/repo/tabs/mod.json b/repo/tabs/mod.json
index e299d9f..f58fb62 100644
--- a/repo/tabs/mod.json
+++ b/repo/tabs/mod.json
@@ -20,6 +20,7 @@
"client": ["client.mjs"],
"electron": [
{ "source": "main.cjs", "target": "main/main.js" },
+ { "source": "systemMenu.cjs", "target": "main/systemMenu.js" },
{ "source": "rendererIndex.cjs", "target": "renderer/index.js" }
]
},
@@ -40,7 +41,7 @@
"type": "select",
"key": "select_modifier",
"label": "tab select modifier",
- "tooltip": "**usage: Modifier+1 to Modifier+9, Modifier+ArrowLeft, Modifier+ArrowRight and Modifier+ link click**",
+ "tooltip": "**usage: Modifier+1 to Modifier+9, Modifier+ArrowLeft and Modifier+ArrowRight**",
"values": [
"Alt",
"Command",
diff --git a/repo/tabs/rendererIndex.cjs b/repo/tabs/rendererIndex.cjs
index b5f4691..931d84c 100644
--- a/repo/tabs/rendererIndex.cjs
+++ b/repo/tabs/rendererIndex.cjs
@@ -29,28 +29,26 @@ module.exports = async function (api, db, __exports, __eval) {
notionUrl: url.parse(window.location.href, true).query.path,
cancelAnimation: true,
});
- $newTab.addEventListener('click', () => {
- new Tab($tabs, $root);
- });
+ $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();
});
+ electron.ipcRenderer.on(
+ 'notion-enhancer:open-tab',
+ (event, opts) => new Tab($tabs, $root, opts)
+ );
let $draggedTab;
- const getDragTarget = ($el) => {
+ const $dragIndicator = web.html``,
+ 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;
},
- clearDragStatus = () => {
- document
- .querySelectorAll('.dragged-over')
- .forEach(($el) => $el.classList.remove('dragged-over'));
- },
resetDraggedTabs = () => {
if ($draggedTab) {
- clearDragStatus();
+ $dragIndicator.remove();
$draggedTab.style.opacity = '';
$draggedTab = undefined;
}
@@ -73,8 +71,11 @@ module.exports = async function (api, db, __exports, __eval) {
$header.addEventListener('dragover', (event) => {
const $target = getDragTarget(event.target);
if ($target) {
- clearDragStatus();
- $target.classList.add('dragged-over');
+ if ($target.matches('#tabs')) {
+ $target.after($dragIndicator);
+ } else if ($target.matches('#tabs > :first-child')) {
+ $tabs.before($dragIndicator);
+ } else $target.before($dragIndicator);
event.preventDefault();
}
});
diff --git a/repo/tabs/systemMenu.cjs b/repo/tabs/systemMenu.cjs
new file mode 100644
index 0000000..9460cff
--- /dev/null
+++ b/repo/tabs/systemMenu.cjs
@@ -0,0 +1,20 @@
+/**
+ * notion-enhancer: tabs
+ * (c) 2021 dragonwocky (https://dragonwocky.me/)
+ * (https://notion-enhancer.github.io/) under the MIT license
+ */
+
+'use strict';
+
+module.exports = async function ({}, db, __exports, __eval) {
+ const notionSetupSystemMenu = __exports.setupSystemMenu;
+ __exports.setupSystemMenu = (locale) => {
+ const { Menu } = require('electron'),
+ template = notionSetupSystemMenu(locale);
+ for (const category of template) {
+ category.submenu = category.submenu.filter((item) => item.role !== 'close');
+ }
+ Menu.setApplicationMenu(Menu.buildFromTemplate(template));
+ return template;
+ };
+};
diff --git a/repo/tabs/tab.cjs b/repo/tabs/tab.cjs
index 801aeaa..25642c2 100644
--- a/repo/tabs/tab.cjs
+++ b/repo/tabs/tab.cjs
@@ -48,8 +48,8 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, tabCache
);
constructor(
- $tabs,
- $root,
+ $tabList,
+ $tabContainer,
{
notionUrl = 'notion://www.notion.so/',
cancelAnimation = false,
@@ -57,14 +57,17 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, tabCache
title = 'notion.so',
} = {}
) {
+ this.$tabList = $tabList;
+ this.$tabContainer = $tabContainer;
+
this.$notion.src = notionUrl;
this.$tabTitle.innerText = title;
this.setIcon(icon);
tabCache.set(this.$tab.id, this);
- web.render($tabs, this.$tab);
- web.render($root, this.$search);
- web.render($root, this.$notion);
+ web.render($tabList, this.$tab);
+ web.render($tabContainer, this.$search);
+ web.render($tabContainer, this.$notion);
electronWindow.on('focus', () => {
if (focusedTab === this) this.$notion.focus();
});
@@ -192,6 +195,16 @@ module.exports = async function ({ components, env, web, fmt, fs }, db, tabCache
this.$tabTitle.innerText = title;
});
fromNotion('notion-enhancer:set-tab-icon', (icon) => this.setIcon(icon));
+
+ fromNotion(
+ 'notion-enhancer:new-tab',
+ () => new this.constructor(this.$tabList, this.$tabContainer)
+ );
+ fromNotion('notion-enhancer:close-tab', () => this.close());
+ fromNotion('notion-enhancer:select-tab', (i) => {
+ const $tab = i === 9 ? this.$tabList.lastElementChild : this.$tabList.children[i - 1];
+ if ($tab) $tab.click();
+ });
}
#firstQuery = true;
diff --git a/repo/tabs/tabs.css b/repo/tabs/tabs.css
index bc85ee3..f887c35 100644
--- a/repo/tabs/tabs.css
+++ b/repo/tabs/tabs.css
@@ -46,7 +46,7 @@ header {
width: 14em;
max-width: 14em;
overflow: hidden;
- padding: 0.4em 0.6em 0.4em 0.6em;
+ padding: 0.4em 0.6em;
color: var(--theme--text_secondary);
background: var(--theme--bg);
@@ -61,26 +61,28 @@ header {
.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);
+
+.drag-indicator {
+ z-index: 1;
+ width: 0.125em;
+ margin: 0 -0.0625em;
+ background: var(--theme--accent_blue-selection);
}
.tab-title {
white-space: nowrap;
overflow: hidden;
+ margin-right: 0.25em;
}
.tab-icon {
- font-size: 0.875em;
margin-right: 0.375em;
+ flex-shrink: 0;
}
.tab-icon[style*='background'] {
width: 0.875em;
height: 0.875em;
align-self: center;
- margin: 0 0.5em 2px 0;
+ margin-right: 0.5em;
}
.tab-icon:not([style*='background']):empty,
.tab-icon + svg {
@@ -88,6 +90,7 @@ header {
}
.tab-icon:not([style*='background']):empty + svg {
/* placeholder icon */
+ flex-shrink: 0;
display: inline-block;
margin: 1px 0.5em 0 0;
width: 1.125em;
@@ -110,9 +113,9 @@ header {
width: 1.25em;
padding: 0 0.25px 0 0;
+ align-self: center;
border: none;
background: transparent;
- font-size: 0.875em;
-webkit-app-region: no-drag;
}
.new-tab svg,
@@ -134,7 +137,6 @@ header {
}
.new-tab {
- align-self: center;
margin: 0 3em 0 0.375em;
}
.tab-close {
@@ -165,9 +167,18 @@ header {
[data-tab-style='traditional tabbed'] .new-tab {
margin-bottom: -0.25em;
}
+[data-tab-style='rectangular'] .tab-close,
+[data-tab-style='traditional tabbed'] .tab-close {
+ align-self: auto;
+}
+[data-tab-style='rectangular'] .drag-indicator,
+[data-tab-style='traditional tabbed'] .drag-indicator,
[data-tab-style='rectangular'] #tabs {
margin-bottom: -0.5em;
}
+[data-tab-style='rectangular'] .tab {
+ padding: 0.6em 0.6em 0.8em 0.6em;
+}
[data-tab-style='traditional tabbed'] header {
padding-top: 0.6875em;
}
@@ -177,7 +188,7 @@ header {
[data-tab-style='traditional tabbed'] .tab {
border-top-left-radius: 0.875em;
border-top-right-radius: 0.875em;
- padding: 0.6em 0.6em 0.4em 0.6em;
+ padding: 0.6em;
}
[data-tab-style='bubble'] .tab {
border-radius: 0.375em;
@@ -185,13 +196,16 @@ header {
[data-tab-style='bubble'] .tab:not(:first-child) {
margin-left: 0.5em;
}
+[data-tab-style='bubble'] .drag-indicator {
+ margin: 0 -0.3125em 0 0.1875em;
+}
+[data-tab-style='bubble'] .drag-indicator:first-child {
+ margin: 0 0.187em 0 -0.312em;
+}
[data-tab-style='compact'] header {
padding: 0;
font-size: 14px;
}
-[data-tab-style='compact'] .tab-close {
- align-self: center;
-}
[data-tab-style='compact'] #window-actions {
transform: scale(0.8);
margin-right: -0.35em;