zotero/chrome/content/zotero/tabs.js
2024-08-26 07:57:17 -04:00

1153 lines
37 KiB
JavaScript

/*
***** BEGIN LICENSE BLOCK *****
Copyright © 2020 Corporation for Digital Scholarship
Vienna, Virginia, USA
https://www.zotero.org
This file is part of Zotero.
Zotero is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Zotero is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
***** END LICENSE BLOCK *****
*/
'use strict';
// Using 'import' breaks hooks
var React = require('react');
var ReactDOM = require('react-dom');
import TabBar from 'components/tabBar';
import { CSSIcon, CSSItemTypeIcon } from 'components/icons';
// Reduce loaded tabs limit if the system has 8 GB or less memory.
// TODO: Revise this after upgrading to Zotero 7
const MAX_LOADED_TABS = Services.sysinfo.getProperty("memsize") / 1024 / 1024 / 1024 <= 8 ? 3 : 5;
const UNLOAD_UNUSED_AFTER = 86400; // 24h
var Zotero_Tabs = new function () {
Object.defineProperty(this, 'selectedID', {
get: () => this._selectedID
});
Object.defineProperty(this, 'selectedType', {
get: () => this._getTab(this._selectedID).tab.type
});
Object.defineProperty(this, 'selectedIndex', {
get: () => this._getTab(this._selectedID).tabIndex
});
Object.defineProperty(this, 'deck', {
get: () => document.getElementById('tabs-deck')
});
Object.defineProperty(this, 'numTabs', {
get: () => this._tabs.length
});
Object.defineProperty(this, 'focusOptions', {
get: () => this._focusOptions
});
Object.defineProperty(this, 'tabsMenuList', {
get: () => document.getElementById('zotero-tabs-menu-list')
});
Object.defineProperty(this, 'tabsMenuPanel', {
get: () => document.getElementById('zotero-tabs-menu-panel')
});
this._tabBarRef = React.createRef();
this._tabs = [{
id: 'zotero-pane',
type: 'library',
title: '',
data: {}
}];
this._selectedID = 'zotero-pane';
this._prevSelectedID = null;
this._history = [];
this._focusOptions = {};
this._tabsMenuFilter = "";
this._tabsMenuFocusedIndex = 0;
this._tabsMenuIgnoreMouseover = false;
this._unloadInterval = setInterval(() => {
this.unloadUnusedTabs();
}, 60000); // Trigger every minute
this._getTab = function (id) {
var tabIndex = this._tabs.findIndex(tab => tab.id == id);
return { tab: this._tabs[tabIndex], tabIndex };
};
this._update = function () {
// Go through all tabs and try to save their icons to tab.data
for (let tab of this._tabs) {
// Find the icon for the library tab
if (tab.id === 'zotero-pane') {
let index = ZoteroPane.collectionsView?.selection?.focused;
if (typeof index !== 'undefined' && ZoteroPane.collectionsView.getRow(index)) {
let iconName = ZoteroPane.collectionsView.getIconName(index);
tab.data.icon = iconName;
}
}
else if (!tab.data.icon) {
// Try to fetch the icon for the reader tab
try {
let item = Zotero.Items.get(tab.data.itemID);
tab.data.icon = item.getItemTypeIconName(true);
}
catch (e) {
// item might not yet be loaded, we will get the right icon on the next update
}
}
}
this._tabBarRef.current.setTabs(this._tabs.map((tab) => {
return {
id: tab.id,
type: tab.type,
title: tab.title,
renderTitle: tab.type === 'reader' || tab.type === 'reader-unloaded',
selected: tab.id == this._selectedID,
isItemType: tab.id !== 'zotero-pane',
icon: tab.data?.icon || null
};
}));
// Disable File > Close menuitem if multiple tabs are open
const multipleTabsOpen = this._tabs.length > 1;
document.getElementById('cmd_close').setAttribute('disabled', multipleTabsOpen);
var { tab } = this._getTab(this._selectedID);
if (!tab) {
return;
}
document.title = (tab.title.length ? tab.title + ' - ' : '') + Zotero.appName;
if (this.isTabsMenuVisible()) {
this.refreshTabsMenuList();
if (document.activeElement.id !== "zotero-tabs-menu-filter") {
focusTabsMenuEntry();
}
}
// Disable tabs menu button when no reader tabs are present
document.getElementById("zotero-tb-tabs-menu").disabled = this._tabs.length == 1;
// Close tabs menu if all tabs are closed
if (this._tabs.length == 1 && this.isTabsMenuVisible()) {
this.tabsMenuPanel.hidePopup();
}
};
this.getTabIDByItemID = function (itemID) {
let tab = this._tabs.find(tab => tab.data && tab.data.itemID === itemID);
return tab && tab.id;
};
this.setSecondViewState = function (tabID, state) {
let { tab } = this._getTab(tabID);
tab.data.secondViewState = state;
Zotero.Session.debounceSave();
};
this.init = function () {
ReactDOM.createRoot(document.getElementById('tab-bar-container')).render(
<TabBar
ref={this._tabBarRef}
onTabSelect={this.select.bind(this)}
onTabMove={this.move.bind(this)}
onTabClose={this.close.bind(this)}
onContextMenu={this._openMenu.bind(this)}
refocusReader={this.refocusReader.bind(this)}
onLoad={this._update.bind(this)}
/>
);
};
this.getState = function () {
return this._tabs.map((tab) => {
let type = tab.type;
if (type === 'reader-unloaded') {
type = 'reader';
}
var o = {
type,
title: tab.title,
timeUnselected: tab.timeUnselected
};
if (tab.data) {
o.data = tab.data;
}
if (tab.id == this._selectedID) {
o.selected = true;
}
return o;
});
};
this.restoreState = function (tabs) {
for (let i = 0; i < tabs.length; i++) {
let tab = tabs[i];
if (tab.type === 'library') {
this.rename('zotero-pane', tab.title);
// At first, library tab is added without the icon data. We set it here once we know what it is
let libraryTab = this._getTab('zotero-pane');
libraryTab.tab.data = tab.data || {};
}
else if (tab.type === 'reader') {
if (Zotero.Items.exists(tab.data.itemID)) {
this.add({
type: 'reader-unloaded',
title: tab.title,
index: i,
data: tab.data,
select: tab.selected
});
}
}
}
// Unset the previously selected tab id, because it was set when restoring tabs
this._prevSelectedID = null;
};
/**
* Add a new tab
*
* @param {String} type
* @param {String} title
* @param {String} data - Extra data about the tab to pass to notifier and session
* @param {Integer} index
* @param {Boolean} select
* @param {Function} onClose
* @return {{ id: string, container: XULElement}} id - tab id, container - a new tab container created in the deck
*/
this.add = function ({ id, type, data, title, index, select, onClose, preventJumpback }) {
if (typeof type != 'string') {
}
if (typeof title != 'string') {
throw new Error(`'title' should be a string (was ${typeof title})`);
}
if (index !== undefined && (!Number.isInteger(index) || index < 1)) {
throw new Error(`'index' should be an integer > 0 (was ${index} (${typeof index})`);
}
if (onClose !== undefined && typeof onClose != 'function') {
throw new Error(`'onClose' should be a function (was ${typeof onClose})`);
}
id = id || 'tab-' + Zotero.Utilities.randomString();
var container = document.createXULElement('vbox');
container.id = id;
this.deck.appendChild(container);
var tab = { id, type, title, data, onClose };
index = index || this._tabs.length;
this._tabs.splice(index, 0, tab);
this._update();
Zotero.Notifier.trigger('add', 'tab', [id], { [id]: Object.assign({}, data, { type }) }, true);
if (select) {
let previousID = this._selectedID;
this.select(id);
if (!preventJumpback) {
this._prevSelectedID = previousID;
}
}
return { id, container };
};
/**
* Set a new tab title
*
* @param {String} id
* @param {String} title
*/
this.rename = function (id, title) {
if (typeof title != 'string') {
throw new Error(`'title' should be a string (was ${typeof title})`);
}
var { tab } = this._getTab(id);
if (!tab) {
return;
}
tab.title = title;
this._update();
};
/**
* Close tabs
*
* @param {String|Array<String>|undefined} ids One or more ids, or empty for the current tab
*/
this.close = function (ids) {
if (!ids) {
ids = [this._selectedID];
}
else if (!Array.isArray(ids)) {
ids = [ids];
}
if (ids.includes('zotero-pane')) {
throw new Error('Library tab cannot be closed');
}
var historyEntry = [];
var closedIDs = [];
var tmpTabs = this._tabs.slice();
for (var id of ids) {
let { tab, tabIndex } = this._getTab(id);
if (!tab) {
continue;
}
if (tab.id == this._selectedID) {
this.select(this._prevSelectedID || (this._tabs[tabIndex + 1] || this._tabs[tabIndex - 1]).id);
}
if (tab.id == this._prevSelectedID) {
this._prevSelectedID = null;
}
tabIndex = this._tabs.findIndex(x => x.id === id);
this._tabs.splice(tabIndex, 1);
if (tab.onClose) {
tab.onClose();
}
historyEntry.push({ index: tmpTabs.indexOf(tab), data: tab.data });
closedIDs.push(id);
setTimeout(() => {
document.getElementById(tab.id).remove();
// For unknown reason fx102, unlike 60, sometimes doesn't automatically update selected index
let selectedIndex = Array.from(this.deck.children).findIndex(x => x.id == this._selectedID);
if (this.deck.selectedIndex !== selectedIndex) {
this.deck.selectedIndex = selectedIndex;
}
});
}
this._history.push(historyEntry);
Zotero.Notifier.trigger('close', 'tab', [closedIDs], true);
this._update();
};
/**
* Close all tabs except the first one
*/
this.closeAll = function () {
this.close(this._tabs.slice(1).map(x => x.id));
};
/**
* Undo tabs closing
*/
this.undoClose = async function () {
var historyEntry = this._history.pop();
if (historyEntry) {
let maxIndex = -1;
let openPromises = [];
for (let tab of historyEntry) {
if (Zotero.Items.exists(tab.data.itemID)) {
openPromises.push(Zotero.Reader.open(tab.data.itemID,
null,
{
tabIndex: tab.index,
openInBackground: true
}
));
if (tab.index > maxIndex) {
maxIndex = tab.index;
}
}
}
await Promise.all(openPromises);
// Select last reopened tab
if (maxIndex > -1) {
this.jump(maxIndex);
}
}
};
/**
* Move a tab to the specified index
*
* @param {String} id
* @param {Integer} newIndex
*/
this.move = function (id, newIndex) {
if (!Number.isInteger(newIndex) || newIndex < 1) {
throw new Error(`'newIndex' should be an interger > 0 (was ${newIndex} (${typeof newIndex})`);
}
var { tab, tabIndex } = this._getTab(id);
if (tabIndex == 0) {
throw new Error('Library tab cannot be moved');
}
if (!tab || tabIndex == newIndex) {
return;
}
if (newIndex > tabIndex) {
newIndex--;
}
this._tabs.splice(tabIndex, 1);
this._tabs.splice(newIndex, 0, tab);
this._update();
};
/**
* Select a tab
*
* @param {String} id
* @param {Boolean} reopening
*/
this.select = function (id, reopening, options = {}) {
var { tab, tabIndex } = this._getTab(id);
// Move focus to the last focused element of zoteroPane if any or itemTree otherwise
let focusZoteroPane = () => {
if (tab.id !== 'zotero-pane') return;
if (options.focusElementID) {
tab.lastFocusedElement = document.getElementById(options.focusElementID);
}
// Small delay to make sure the focus does not remain on the actual
// tab after mouse click
setTimeout(() => {
if (tab.lastFocusedElement) {
tab.lastFocusedElement.focus();
}
if (document.activeElement !== tab.lastFocusedElement) {
ZoteroPane_Local.itemsView.focus();
}
tab.lastFocusedElement = null;
});
};
if (!tab || tab.id === this._selectedID) {
// Focus on reader or zotero pane when keepTabFocused is explicitly false
// E.g. when a tab is selected via Space or Enter
if (options.keepTabFocused === false && tab?.id === this._selectedID) {
var reader = Zotero.Reader.getByTabID(this._selectedID);
if (reader) {
reader.focus();
}
if (tab.id == 'zotero-pane') {
focusZoteroPane();
}
}
return;
}
let selectedTab;
if (this._selectedID) {
selectedTab = this._getTab(this._selectedID).tab;
if (selectedTab) {
selectedTab.timeUnselected = Zotero.Date.getUnixTimestamp();
}
}
// If the last focus data was recorded for a different item, discard it
if (!this._focusOptions.itemID || this._focusOptions.itemID != tab?.data?.itemID) {
this._focusOptions = {};
}
// Save focus option for this item to tell reader and contextPane how to handle focus
if (Object.keys(options).length && selectedTab) {
this._focusOptions.keepTabFocused = !!options.keepTabFocused;
this._focusOptions.itemID = tab?.data?.itemID;
}
if (this._selectedID === 'zotero-pane'
&& !document.activeElement.classList.contains("tab")
&& document.activeElement.tagName !== 'window') {
// never return focus to another tab or <window>
selectedTab.lastFocusedElement = document.activeElement;
}
if (tab.type === 'reader-unloaded') {
tab.type = "reader-loading";
// Make sure the loading message is displayed first.
// Then, open reader and hide the loading message once it has loaded.
ZoteroContextPane.showLoadingMessage(true);
let hideMessageWhenReaderLoaded = async () => {
let reader = await Zotero.Reader.open(tab.data.itemID, options && options.location, {
tabID: tab.id,
title: tab.title,
tabIndex,
allowDuplicate: true,
secondViewState: tab.data.secondViewState,
preventJumpback: true
});
await reader._initPromise;
ZoteroContextPane.showLoadingMessage(false);
};
hideMessageWhenReaderLoaded();
}
this._prevSelectedID = reopening ? this._selectedID : null;
this._selectedID = id;
this.deck.selectedIndex = Array.from(this.deck.children).findIndex(x => x.id == id);
this._update();
Zotero.Notifier.trigger('select', 'tab', [tab.id], { [tab.id]: { type: tab.type } }, true);
if (tab.id === 'zotero-pane' && (options.keepTabFocused !== true)) {
focusZoteroPane();
}
let tabNode = document.querySelector(`#tab-bar-container .tab[data-id="${tab.id}"]`);
if (this._focusOptions.keepTabFocused && document.activeElement.getAttribute('data-id') != tabNode.getAttribute('data-id')) {
// Keep focus on the currently selected tab during keyboard navigation
if (tab.id == 'zotero-pane') {
// Since there is more than one zotero-pane tab (pinned and not pinned),
// use moveFocus() to focus on the visible one
this.moveFocus('first');
}
else {
tabNode.focus();
}
}
tab.timeSelected = Zotero.Date.getUnixTimestamp();
// Without `setTimeout` the tab closing that happens in `unloadUnusedTabs` results in
// tabs deck selection index bigger than the deck children count. It feels like something
// isn't update synchronously
setTimeout(() => this.unloadUnusedTabs());
};
this.unload = function (id) {
var { tab, tabIndex } = this._getTab(id);
if (!tab || tab.id === this._selectedID || tab.type !== 'reader') {
return;
}
this.close(tab.id);
this.add({
id: tab.id,
type: 'reader-unloaded',
title: tab.title,
index: tabIndex,
data: tab.data
});
};
// Mark a tab as loaded
this.markAsLoaded = function (id) {
let { tab } = this._getTab(id);
if (!tab || tab.type == "reader") return;
let prevType = tab.type;
tab.type = "reader";
Zotero.Notifier.trigger("load", "tab", [id], { [id]: Object.assign({}, tab, { prevType }) }, true);
};
this.unloadUnusedTabs = function () {
for (let tab of this._tabs) {
if (Zotero.Date.getUnixTimestamp() - tab.timeUnselected > UNLOAD_UNUSED_AFTER) {
this.unload(tab.id);
}
}
let tabs = this._tabs.slice().filter(x => x.type === 'reader');
tabs.sort((a, b) => b.timeUnselected - a.timeUnselected);
tabs = tabs.slice(MAX_LOADED_TABS);
for (let tab of tabs) {
this.unload(tab.id);
}
};
/**
* Select the previous tab (closer to the library tab)
*/
this.selectPrev = function (options) {
var { tabIndex } = this._getTab(this._selectedID);
this.select((this._tabs[tabIndex - 1] || this._tabs[this._tabs.length - 1]).id, false, options || {});
};
/**
* Select the next tab (farther to the library tab)
*/
this.selectNext = function (options) {
var { tabIndex } = this._getTab(this._selectedID);
this.select((this._tabs[tabIndex + 1] || this._tabs[0]).id, false, options || {});
};
/**
* Select the last tab
*/
this.selectLast = function () {
this.select(this._tabs[this._tabs.length - 1].id);
};
/**
* Return focus into the reader of the selected tab.
* Required to move focus from the tab into the reader after drag.
*/
this.refocusReader = function () {
var reader = Zotero.Reader.getByTabID(this._selectedID);
if (!reader) return;
setTimeout(() => {
reader.focus();
});
};
/**
* Moves focus to a tab in the specified direction.
* @param {String} direction. "first", "last", "previous", "next", or "current"
* If document.activeElement is a tab, "previous" or "next" direction moves focus from that tab.
* Otherwise, focus is moved in the given direction from the currently selected tab.
*/
this.moveFocus = function (direction) {
let focusedTabID = document.activeElement.getAttribute('data-id');
var { tabIndex } = this._getTab(this._selectedID);
let tabIndexToFocus = null;
if (direction === "last") {
tabIndexToFocus = this._tabs.length - 1;
}
else if (direction == "first") {
tabIndexToFocus = 0;
}
else if (direction == "current") {
tabIndexToFocus = tabIndex;
}
else {
let focusedTabIndex = this._tabs.findIndex(tab => tab.id === focusedTabID);
// If the currently focused element is not a tab, use tab that is selected
if (focusedTabIndex === -1) {
focusedTabIndex = tabIndex;
}
switch (direction) {
case "previous":
tabIndexToFocus = focusedTabIndex > 0 ? focusedTabIndex - 1 : null;
break;
case "next":
tabIndexToFocus = focusedTabIndex < this._tabs.length - 1 ? focusedTabIndex + 1 : null;
break;
default:
throw new Error(`${direction} is an invalid direction.`);
}
}
if (tabIndexToFocus !== null) {
const nextTab = this._tabs[tabIndexToFocus];
// There may be duplicate tabs - in normal tab array and in pinned tabs
// Go through all candidates and try to focus the visible one
let candidates = document.querySelectorAll(`#tab-bar-container .tab[data-id="${nextTab.id}"]`);
for (let node of candidates) {
node.focus();
// Visible tab was found and focused
if (document.activeElement == node) return;
}
}
};
/**
* Jump to the tab at a particular index. If the index points beyond the array, jump to the last
* tab.
*
* @param {Integer} index
*/
this.jump = function (index) {
this.select(this._tabs[Math.min(index, this._tabs.length - 1)].id);
};
this._openMenu = function (x, y, id) {
var { tab, tabIndex } = this._getTab(id);
let menuitem;
let popup = document.createXULElement('menupopup');
document.querySelector('popupset').appendChild(popup);
popup.addEventListener('popuphidden', function (event) {
if (event.target === popup) {
popup.remove();
}
});
if (id !== 'zotero-pane') {
// Show in library
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('general.showInLibrary'));
menuitem.addEventListener('command', () => {
let { tab } = this._getTab(id);
if (tab && (tab.type === 'reader' || tab.type === 'reader-unloaded')) {
let itemID = tab.data.itemID;
let item = Zotero.Items.get(itemID);
if (item && item.parentItemID) {
itemID = item.parentItemID;
}
ZoteroPane_Local.selectItem(itemID);
}
});
popup.appendChild(menuitem);
// Move tab
let menu = document.createXULElement('menu');
menu.setAttribute('label', Zotero.getString('tabs.move'));
let menupopup = document.createXULElement('menupopup');
menu.append(menupopup);
popup.appendChild(menu);
// Move to start
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('tabs.moveToStart'));
menuitem.setAttribute('disabled', tabIndex == 1);
menuitem.addEventListener('command', () => {
this.move(id, 1);
});
menupopup.appendChild(menuitem);
// Move to end
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('tabs.moveToEnd'));
menuitem.setAttribute('disabled', tabIndex == this._tabs.length - 1);
menuitem.addEventListener('command', () => {
this.move(id, this._tabs.length);
});
menupopup.appendChild(menuitem);
// Move to new window
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('tabs.moveToWindow'));
menuitem.setAttribute('disabled', false);
menuitem.addEventListener('command', () => {
let { tab } = this._getTab(id);
if (tab && (tab.type === 'reader' || tab.type === 'reader-unloaded')) {
this.close(id);
let { itemID, secondViewState } = tab.data;
Zotero.Reader.open(itemID, null, { openInWindow: true, secondViewState });
}
});
menupopup.appendChild(menuitem);
// Duplicate tab
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('tabs.duplicate'));
menuitem.addEventListener('command', () => {
if (tab.data.itemID) {
tabIndex++;
let { secondViewState } = tab.data;
Zotero.Reader.open(tab.data.itemID, null, { tabIndex, allowDuplicate: true, secondViewState });
}
});
popup.appendChild(menuitem);
// Separator
popup.appendChild(document.createXULElement('menuseparator'));
}
// Close
if (id != 'zotero-pane') {
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('general.close'));
menuitem.addEventListener('command', () => {
this.close(id);
});
popup.appendChild(menuitem);
}
// Close other tabs
if (!(this._tabs.length == 2 && id != 'zotero-pane')) {
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute('label', Zotero.getString('tabs.closeOther'));
menuitem.addEventListener('command', () => {
this.close(this._tabs.slice(1).filter(x => x.id != id).map(x => x.id));
});
popup.appendChild(menuitem);
}
// Undo close
menuitem = document.createXULElement('menuitem');
menuitem.setAttribute(
'label',
Zotero.getString(
'tabs.undoClose',
[],
// If not disabled, show proper plural for tabs to reopen
this._history.length ? this._history[this._history.length - 1].length : 1
)
);
menuitem.setAttribute('disabled', !this._history.length);
menuitem.addEventListener('command', () => {
this.undoClose();
});
popup.appendChild(menuitem);
popup.openPopupAtScreen(x, y, true);
};
// Used to move focus back to itemTree or contextPane from the tabs.
this.focusWrapAround = function () {
// Focus the last field of contextPane when reader is opened
if (Zotero_Tabs.selectedIndex > 0) {
Services.focus.moveFocus(window, document.getElementById("zotero-context-pane-sidenav"),
Services.focus.MOVEFOCUS_BACKWARD, 0);
return;
}
// Focus the last field of itemPane
// We do that by moving focus backwards from the element following the pane, because Services.focus doesn't
// support MOVEFOCUS_LAST on subtrees
Services.focus.moveFocus(window, document.getElementById("zotero-context-splitter"),
Services.focus.MOVEFOCUS_BACKWARD, 0);
};
/**
* @param {title} String - Tab's title
* @returns <description> with bold substrings of title matching this._tabsMenuFilter
*/
let createTabsMenuLabel = (title) => {
let desc = document.createElement('label');
let regex = new RegExp(`(${Zotero.Utilities.quotemeta(this._tabsMenuFilter)})`, 'gi');
let matches = title.matchAll(regex);
let lastIndex = 0;
for (let match of matches) {
if (match.index > lastIndex) {
// Add preceding text
desc.appendChild(document.createTextNode(title.substring(lastIndex, match.index)));
}
// Add matched text wrapped in <b>
if (match[0]) {
let b = document.createElement('b');
b.textContent = match[0];
desc.appendChild(b);
}
lastIndex = match.index + match[0].length;
}
if (lastIndex < title.length) {
// Add remaining text
desc.appendChild(document.createTextNode(title.substring(lastIndex)));
}
return desc;
};
this.isTabsMenuVisible = () => {
return ['showing', 'open'].includes(this.tabsMenuPanel.state);
};
/**
* Create the list of opened tabs in tabs menu.
*/
this.refreshTabsMenuList = () => {
// Empty existing nodes
this.tabsMenuList.replaceChildren();
this._tabsMenuFocusedIndex = 0;
let index = 1;
for (let tab of this._tabs) {
// Skip tabs whose title wasn't added yet
if (tab.title == "") {
continue;
}
// Filter tabs that do not match the filter
if (!tab.title.toLowerCase().includes(this._tabsMenuFilter)) {
continue;
}
// Top-level entry of the opened tabs array
let row = document.createElement('div');
let rowIndex = this._tabs.findIndex(x => x.id === tab.id);
row.classList = "row";
row.setAttribute("index", rowIndex);
row.setAttribute("draggable", true);
// Title of the tab
let tabName = document.createElement('div');
tabName.setAttribute('flex', '1');
tabName.setAttribute('class', 'zotero-tabs-menu-entry title');
tabName.setAttribute('tabindex', `${index++}`);
tabName.setAttribute('aria-label', tab.title);
tabName.setAttribute('title', tab.title);
// Cross button to close a tab
let closeButton = document.createElement('div');
closeButton.className = "zotero-tabs-menu-entry close";
let closeIcon = document.createElement('span');
closeIcon.setAttribute('class', 'icon icon-css icon-x-8 icon-16');
closeButton.setAttribute('data-l10n-id', 'zotero-tabs-menu-close-button');
closeButton.appendChild(closeIcon);
closeButton.addEventListener("click", () => {
// Keep the focus on the cross at the same spot
if (this._tabsMenuFocusedIndex == this.tabsMenuList.childElementCount * 2) {
this._tabsMenuFocusedIndex = Math.max(this._tabsMenuFocusedIndex - 2, 0);
}
this.close(tab.id);
});
// Library tab has no close button
if (tab.id == "zotero-pane") {
closeButton.hidden = true;
}
closeButton.setAttribute('tabindex', `${index++}`);
// Item type icon
let span = document.createElement("span");
span.className = "icon icon-css tab-icon";
if (tab.id == 'zotero-pane') {
// Determine which icon from the collection view rows to use (same as in _update())
let index = ZoteroPane.collectionsView?.selection?.focused;
if (typeof index !== 'undefined' && ZoteroPane.collectionsView.getRow(index)) {
let iconName = ZoteroPane.collectionsView.getIconName(index);
span.classList.add(`icon-${iconName}`);
}
}
else {
span.classList.add("icon-item-type");
let item = Zotero.Items.get(tab.data.itemID);
let dataTypeLabel = item.getItemTypeIconName(true);
span.setAttribute("data-item-type", dataTypeLabel);
}
tabName.appendChild(span);
// Actual label with bolded substrings matching the filter
let tabLabel = createTabsMenuLabel(tab.title, this._tabsMenuFilter);
tabName.appendChild(tabLabel);
// Selected tab is bold
if (tab.id == this._selectedID) {
tabName.classList.add('selected');
}
// Onclick, go to selected tab + close popup
tabName.addEventListener("click", () => {
this.tabsMenuPanel.hidePopup();
this.select(tab.id);
});
row.appendChild(tabName);
row.appendChild(closeButton);
row.addEventListener("dragstart", (e) => {
// No drag-drop on the cross button or the library tab
if (tab.id == 'zotero-pane' || e.target.classList.contains("close")) {
e.preventDefault();
e.stopPropagation();
return;
}
e.dataTransfer.setData('zotero/tab', tab.id);
setTimeout(() => {
row.classList.remove("hover");
row.setAttribute("id", "zotero-tabs-menu-dragged");
});
});
row.addEventListener('dragover', (e) => {
e.preventDefault();
let tabId = e.dataTransfer.getData("zotero/tab");
if (!tabId || tab.id == "zotero-pane") {
return false;
}
if (row.getAttribute("id") == "zotero-tabs-menu-dragged") {
return true;
}
let placeholder = document.getElementById("zotero-tabs-menu-dragged");
if (row.previousSibling?.id == placeholder.id) {
// If the placeholder exists before the row, swap the placeholder and the row
row.parentNode.insertBefore(row, placeholder);
placeholder.setAttribute("index", parseInt(row.getAttribute("index")) + 1);
}
else {
// Insert placeholder before the row
row.parentNode.insertBefore(placeholder, row);
placeholder.setAttribute("index", parseInt(row.getAttribute("index")));
}
return false;
});
row.addEventListener('drop', (e) => {
let tabId = e.dataTransfer.getData("zotero/tab");
let rowIndex = parseInt(row.getAttribute("index"));
if (rowIndex == 0) return;
this.move(tabId, rowIndex);
});
row.addEventListener('dragend', (_) => {
// If this.move() wasn't called, just re-render the menu
if (document.getElementById("zotero-tabs-menu-dragged")) {
this.refreshTabsMenuList();
}
});
this.tabsMenuList.appendChild(row);
}
};
this.showTabsMenu = function (button) {
this.tabsMenuPanel.openPopup(button, "after_start", -20, -2, false, false);
};
this.handleTabsMenuHiding = function (event) {
if (event.originalTarget.id != 'zotero-tabs-menu-panel') return;
// Empty out the filter input field
let menuFilter = document.getElementById('zotero-tabs-menu-filter');
menuFilter.value = "";
this._tabsMenuFilter = "";
this.tabsMenuList.closest("panel").style.removeProperty('max-height');
};
this.handleTabsMenuShown = function (_) {
focusTabsMenuEntry(0);
};
this.handleTabsMenuShowing = function (_) {
this.refreshTabsMenuList();
// Make sure that if the menu is very long, there is a small
// gap left between the top/bottom of the menu and the edge of the screen
let valuesAreWithinMargin = (valueOne, valueTwo, margin) => {
return Math.abs(valueOne - valueTwo) <= margin;
};
let panel = document.getElementById("zotero-tabs-menu-panel");
let panelRect = panel.getBoundingClientRect();
const gapBeforeScreenEdge = 25;
let absoluteTabsMenuTop = window.screenY - panelRect.height + panelRect.bottom;
let absoluteTabsMenuBottom = window.screenY + panelRect.height + panelRect.top;
// On windows, getBoundingClientRect does not give us correct top and bottom values
// until popupshown, so instead use the anchor's position
if (Zotero.isWin) {
let anchor = document.getElementById("zotero-tb-tabs-menu");
let anchorRect = anchor.getBoundingClientRect();
absoluteTabsMenuTop = window.screenY - panelRect.height + anchorRect.top;
absoluteTabsMenuBottom = window.screenY + panelRect.height + anchorRect.bottom;
}
// screen.availTop is not always right on Linux, so ignore it
let availableTop = Zotero.isLinux ? 0 : screen.availTop;
// Check if the end of the tabs menu is close to the edge of the screen
let atTopScreenEdge = valuesAreWithinMargin(absoluteTabsMenuTop, availableTop, gapBeforeScreenEdge);
let atBottomScreenEdge = valuesAreWithinMargin(absoluteTabsMenuBottom, screen.availHeight + availableTop, gapBeforeScreenEdge);
let gap;
// Limit max height of the menu to leave the specified gap till the screen's edge.
// Due to screen.availTop behavior on linux, the menu can go outside of what is supposed
// to be the available screen area, so special treatment for those edge cases.
if (atTopScreenEdge || (Zotero.isLinux && absoluteTabsMenuTop < 0)) {
gap = gapBeforeScreenEdge - (absoluteTabsMenuTop - availableTop);
}
if (atBottomScreenEdge || (Zotero.isLinux && absoluteTabsMenuBottom > screen.availHeight)) {
gap = gapBeforeScreenEdge - (screen.availHeight + availableTop - absoluteTabsMenuBottom);
}
if (gap) {
panel.style.maxHeight = `${panelRect.height - gap}px`;
}
// Try to scroll selected tab into the center
let selectedTab = this.tabsMenuList.querySelector(".selected");
if (selectedTab) {
selectedTab.scrollIntoView({ block: 'center' });
}
};
/**
* Record the value of the filter
*/
this.handleTabsMenuFilterInput = function (_, input) {
if (this._tabsMenuFilter == input.value.toLowerCase()) {
return;
}
this._tabsMenuFilter = input.value.toLowerCase();
this.refreshTabsMenuList();
};
this.resetFocusIndex = (_) => {
this._tabsMenuFocusedIndex = 0;
};
/**
* Focus on the element in the tabs menu with [tabindex=tabIndex] if given
* or [tabindex=this._tabsMenuFocusedIndex] otherwise
*/
let focusTabsMenuEntry = (tabIndex = null) => {
tabIndex = tabIndex !== null ? tabIndex : this._tabsMenuFocusedIndex;
if (tabIndex === null) {
return;
}
var nextTab = this.tabsMenuList.parentElement.querySelector(`[tabindex="${tabIndex}"]`);
if (!nextTab) {
return;
}
this._tabsMenuIgnoreMouseover = true;
this._tabsMenuFocusedIndex = tabIndex;
let hovered = this.tabsMenuList.querySelector(".hover");
if (hovered) {
hovered.classList.remove("hover");
}
nextTab.focus();
// For some reason (likely a mozilla bug),
// a mouseover event fires at the location where the drag event started after the drop.
// To not mark the wrong entry as hovered, ignore mouseover events for a bit after the focus change
setTimeout(() => {
this._tabsMenuIgnoreMouseover = false;
}, 250);
};
/**
* Keyboard navigation within the tabs menu
* - Tab/Shift-Tab moves focus from the input field across tab titles and close buttons
* - Enter from the input field focuses the first tab
* - Enter on a toolbarbutton clicks it
* - ArrowUp/ArrowDown on a toolbarbutton moves focus to the next/previous toolbarbutton of the
* same type (e.g. arrowDown from title focuses the next title)
* - ArrowUp from the first tab or ArrowDown from the last tab focuses the filter field
* - ArrowDown from the filter field focuses the first tab
* - ArrowUp from the filter field focuses the last tab
* - Home/PageUp focuses the filter field
* - End/PageDown focues the last tab title
* - CMD-f will focus the input field
*/
this.handleTabsMenuKeyPress = function (event) {
let tabindex = this._tabsMenuFocusedIndex;
if (event.key == "Tab") {
event.preventDefault();
let isShift = event.shiftKey;
let moveTabIndex = () => tabindex++;
if (isShift) {
moveTabIndex = () => tabindex--;
}
moveTabIndex();
let candidate = this.tabsMenuList.parentElement.querySelector(`[tabindex="${tabindex}"]`);
// If the candidate is hidden (e.g. close button of library tab), skip it
if (candidate && candidate.hidden) {
moveTabIndex();
}
focusTabsMenuEntry(tabindex);
}
else if (["Home", "PageUp"].includes(event.key)) {
event.preventDefault();
focusTabsMenuEntry(0);
}
else if (["End", "PageDown"].includes(event.key)) {
event.preventDefault();
focusTabsMenuEntry(this.tabsMenuList.childElementCount * 2 - 1);
}
else if (["ArrowUp", "ArrowDown"].includes(event.key)) {
event.preventDefault();
let isFirstRow = tabindex <= 2 && tabindex > 0;
// Step over 1 index to jump over close button, unless we move
// from the filter field
let step = tabindex == 0 ? 1 : 2;
if (event.key == "ArrowDown") {
tabindex += step;
}
else {
tabindex -= step;
}
// If the candidate is a disabled element (e.g. close button of the library tab),
// move focus to the element before it
let candidate = this.tabsMenuList.parentElement.querySelector(`[tabindex="${tabindex}"]`);
if (candidate && candidate.disabled) {
tabindex--;
}
if (tabindex <= 0) {
// ArrowUp from the first tab or the first close button focuses the filter field.
// ArrowUp from the filter field focuses the last tab
if (isFirstRow) {
tabindex = 0;
}
else {
tabindex = this.tabsMenuList.childElementCount * 2 - 1;
}
}
// ArrowDown from the bottom focuses the filter field
if (tabindex > this.tabsMenuList.childElementCount * 2) {
tabindex = 0;
}
focusTabsMenuEntry(tabindex);
}
else if (["Enter", " "].includes(event.key)) {
event.preventDefault();
if (event.target.id == "zotero-tabs-menu-filter") {
focusTabsMenuEntry(1);
return;
}
event.target.click();
}
else if (["ArrowLeft", "ArrowRight"].includes(event.key)) {
event.preventDefault();
event.stopPropagation();
}
else if (event.key == "f" && (Zotero.isMac ? event.metaKey : event.ctrlKey)) {
focusTabsMenuEntry(0);
event.preventDefault();
event.stopPropagation();
}
};
};