middle click close tabs + better tab animations

This commit is contained in:
dragonwocky 2020-10-16 17:26:18 +11:00
parent b4b54fdca3
commit a264cbf832
Signed by: dragonwocky
GPG Key ID: C7A48B7846AA706D
3 changed files with 45 additions and 15 deletions

View File

@ -79,7 +79,7 @@ module.exports = (store, __exports) => {
}); });
electron.app.on('before-quit', () => (intended_quit = true)); electron.app.on('before-quit', () => (intended_quit = true));
window.loadURL(__exports.getIndexUrl(relativeUrl)); window.loadURL(__exports.getIndexUrl(relativeUrl));
// window.webContents.openDevTools(); window.webContents.openDevTools();
return window; return window;
}; };
return __exports.createWindow; return __exports.createWindow;

View File

@ -23,7 +23,7 @@
transform: rotate(360deg); transform: rotate(360deg);
} }
} }
@keyframes tabSlideIn { @keyframes tabSlide {
from { from {
width: 0; width: 0;
} }
@ -137,7 +137,11 @@ body,
text-overflow: ellipsis; text-overflow: ellipsis;
} }
#tabs .tab.slideIn span:not(.close) { #tabs .tab.slideIn span:not(.close) {
animation: tabSlideIn 100ms ease-in-out; animation: tabSlide 100ms ease-in-out;
}
#tabs .tab.slideOut {
width: 0;
animation: tabSlide 100ms ease-in-out reverse;
} }
#tabs .tab .close { #tabs .tab .close {
padding: 0 0.35em 0.1em 0.3em; padding: 0 0.35em 0.1em 0.3em;

View File

@ -44,6 +44,7 @@ module.exports = (store, __exports) => {
zoomFactor: 1, zoomFactor: 1,
tabs: new Map([[0, ['notion.so', true]]]), tabs: new Map([[0, ['notion.so', true]]]),
slideIn: new Set(), slideIn: new Set(),
slideOut: new Set(),
}; };
this.$titlebar = null; this.$titlebar = null;
this.$dragging = null; this.$dragging = null;
@ -220,9 +221,20 @@ module.exports = (store, __exports) => {
const list = new Map(this.state.tabs); const list = new Map(this.state.tabs);
while (this.state.tabs.get(id) && this.state.tabs.get(id)[1]) id++; while (this.state.tabs.get(id) && this.state.tabs.get(id)[1]) id++;
list.delete(id); list.delete(id);
this.openTab(id, list, true); this.openTab(id, { state: list, load: true });
} }
openTab(id, state = new Map(this.state.tabs), load) { openTab(
id,
{
state = new Map(this.state.tabs),
slideOut = new Set(this.state.slideOut),
load,
} = {
state: new Map(this.state.tabs),
slideOut: new Set(this.state.slideOut),
load: false,
}
) {
if (!id && id !== 0) { if (!id && id !== 0) {
if (state.get(this.views.current.id)[1]) return; if (state.get(this.views.current.id)[1]) return;
const currentIndex = [...state].findIndex( const currentIndex = [...state].findIndex(
@ -241,6 +253,7 @@ module.exports = (store, __exports) => {
true, true,
]), ]),
slideIn: load ? this.state.slideIn.add(id) : this.state.slideIn, slideIn: load ? this.state.slideIn.add(id) : this.state.slideIn,
slideOut: slideOut,
}, },
async () => { async () => {
this.focusTab(); this.focusTab();
@ -268,13 +281,11 @@ module.exports = (store, __exports) => {
? idToNotionURL(store().default_page) ? idToNotionURL(store().default_page)
: current_src : current_src
); );
setTimeout(() => {
const slideIn = new Set(this.state.slideIn);
slideIn.delete(id);
this.setState({ slideIn });
}, 100);
// this.views.html[id].getWebContents().openDevTools(); // this.views.html[id].getWebContents().openDevTools();
} }
setTimeout(() => {
this.setState({ slideIn: new Set(), slideOut: new Set() });
}, 150);
} }
); );
} }
@ -286,7 +297,7 @@ module.exports = (store, __exports) => {
return electron.remote.getCurrentWindow().close(); return electron.remote.getCurrentWindow().close();
this.openTab( this.openTab(
this.views.current.id === id ? null : this.views.current.id, this.views.current.id === id ? null : this.views.current.id,
list { state: list, slideOut: this.state.slideOut.add(id) }
); );
} }
focusTab() { focusTab() {
@ -297,11 +308,20 @@ module.exports = (store, __exports) => {
for (const id in this.views.loaded) { for (const id in this.views.loaded) {
if (this.views.loaded.hasOwnProperty(id) && this.views.loaded[id]) { if (this.views.loaded.hasOwnProperty(id) && this.views.loaded[id]) {
const selected = const selected =
id == this.views.current.id && this.state.tabs.get(+id); id == this.views.current.id &&
this.state.tabs.get(+id) &&
this.state.tabs.get(+id)[1];
this.views.loaded[id].style.display = selected ? 'flex' : 'none'; this.views.loaded[id].style.display = selected ? 'flex' : 'none';
if (selected) { if (selected) {
this.views.active = this.views.current.id; this.views.active = +id;
this.views.loaded[id].focus(); this.views.loaded[id].focus();
const electronWindow = electron.remote.getCurrentWindow();
if (
electronWindow &&
electronWindow.getTitle() !== this.state.tabs.get(+id)[0]
) {
electronWindow.setTitle(this.state.tabs.get(+id)[0]);
}
} }
} }
} }
@ -628,7 +648,9 @@ module.exports = (store, __exports) => {
'div', 'div',
{ id: 'tabs' }, { id: 'tabs' },
...[...this.state.tabs] ...[...this.state.tabs]
.filter(([id, [title, open]]) => open) .filter(
([id, [title, open]]) => open || this.state.slideOut.has(id)
)
.map(([id, [title, open]]) => .map(([id, [title, open]]) =>
React.createElement( React.createElement(
'button', 'button',
@ -636,12 +658,16 @@ module.exports = (store, __exports) => {
className: className:
'tab' + 'tab' +
(id === this.views.current.id ? ' current' : '') + (id === this.views.current.id ? ' current' : '') +
(this.state.slideIn.has(id) ? ' slideIn' : ''), (this.state.slideIn.has(id) ? ' slideIn' : '') +
(this.state.slideOut.has(id) ? ' slideOut' : ''),
draggable: true, draggable: true,
onClick: (e) => { onClick: (e) => {
if (!e.target.classList.contains('close')) if (!e.target.classList.contains('close'))
this.openTab(id); this.openTab(id);
}, },
onMouseUp: (e) => {
if (window.event.which === 2) this.closeTab(id);
},
ref: ($tab) => { ref: ($tab) => {
this.views.tabs[id] = $tab; this.views.tabs[id] = $tab;
}, },