zotero/chrome/content/zotero/lookup.js
abaevbog 620b35ab78 minor focus tweaks
- after an item is added via "Add item by identifier" panel, end
focus to that item in itemTree.
- restored default behavior when the focus remains visible around an
element when all menupopups open, except for the item type menu in itemBox.
- opening item type menu will still hide the focus-ring
2024-01-24 23:32:23 -05:00

307 lines
8.3 KiB
JavaScript

/*
***** BEGIN LICENSE BLOCK *****
Copyright © 2009-2011 Center for History and New Media
George Mason University, Fairfax, Virginia, USA
http://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 *****
*/
/**
* Handles UI for lookup panel
* @namespace
*/
var Zotero_Lookup = new function () {
/**
* Performs a lookup by DOI, PMID, or ISBN on the given textBox value
* and adds any items it can.
*
* If a childItem is passed, then only one identifier is allowed, the
* child's library/collection information is used and no attachments are
* saved for the parent.
*
* @param textBox {HTMLElement} - Textbox containing identifiers
* @param childItem {Zotero.Item|false} - Child item (optional)
* @param toggleProgress {function} - Callback to toggle progress on/off
* @returns {Promise<Zotero.Item[]>}
*/
this._button = null;
this.addItemsFromIdentifier = async function (textBox, childItem, toggleProgress) {
var identifiers = Zotero.Utilities.extractIdentifiers(textBox.value);
if (!identifiers.length) {
Zotero.alert(
window,
Zotero.getString("lookup.failure.title"),
Zotero.getString("lookup.failureToID.description")
);
return false;
}
else if (childItem && identifiers.length > 1) {
// Only allow one identifier when creating a parent for a child
Zotero.alert(
window,
Zotero.getString("lookup.failure.title"),
Zotero.getString("lookup.failureTooMany.description")
);
return false;
}
var libraryID = false;
var collections = false;
if (childItem) {
libraryID = childItem.libraryID;
collections = childItem.collections;
}
else {
try {
libraryID = ZoteroPane.getSelectedLibraryID();
let collection = ZoteroPane.getSelectedCollection();
collections = collection ? [collection.id] : false;
}
catch (e) {
/** TODO: handle this **/
}
}
toggleProgress(true);
// Group PubMed IDs into batches of 200
//
// Up to 10,000 ids can apparently be passed in a single request, but 200 is the recommended
// limit for GET, which we currently use, and passing batches of 200 seems...fine.
//
// https://www.ncbi.nlm.nih.gov/books/NBK25499/#chapter4.id
if (identifiers.length && identifiers[0].PMID) {
let chunkSize = 200;
let newIdentifiers = [];
for (let i = 0; i < identifiers.length; i += chunkSize) {
newIdentifiers.push({
PMID: identifiers.slice(i, i + chunkSize).map(x => x.PMID)
});
}
identifiers = newIdentifiers;
}
let newItems = [];
for (let identifier of identifiers) {
let translate = new Zotero.Translate.Search();
translate.setIdentifier(identifier);
// be lenient about translators
let translators = await translate.getTranslators();
translate.setTranslator(translators);
try {
newItems.push(...await translate.translate({
libraryID,
collections,
saveAttachments: !childItem
}));
}
// Continue with other ids on failure
catch (e) {
Zotero.logError(e);
}
}
toggleProgress(false);
if (!newItems.length) {
Zotero.alert(
window,
Zotero.getString("lookup.failure.title"),
Zotero.getString("lookup.failure.description")
);
}
// TODO: Give indication if some, but not all failed
return newItems;
};
/**
* Try a lookup and hide popup if successful
*/
this.accept = async function (textBox) {
let newItems = await Zotero_Lookup.addItemsFromIdentifier(
textBox,
false,
on => Zotero_Lookup.setShowProgress(on)
);
if (newItems.length) {
// Send the focus to the item tree after the popup closes
ZoteroPane.lastFocusedElement = null;
document.getElementById("zotero-lookup-panel").hidePopup();
document.getElementById("item-tree-main-default").focus();
}
return false;
};
this.showPanel = function (button) {
var panel = document.getElementById('zotero-lookup-panel');
this._button = button;
if (!button) {
button = document.getElementById("zotero-tb-lookup");
}
panel.openPopup(button, "after_start", 16, -2, false, false);
}
this.onFocusOut = function(event) {
/*
if the lookup popup was triggered by the lookup button,
we want to return there on focus out. So we check
(1) that we came from a button and (2) that
event.relatedTarget === null, i.e. that the user hasn't used
the mouse or keyboard to select something, and focus is leaving
the popup because the popup was hidden/dismissed.
*/
if (this._button && event.relatedTarget === null) {
event.preventDefault();
event.stopPropagation();
this._button.focus();
this._button = null;
}
else {
this._button = null;
}
}
/**
* Focuses the field
*/
this.onShowing = function (event) {
// Ignore context menu
if (event.originalTarget.id != 'zotero-lookup-panel') return;
document.getElementById("zotero-lookup-panel").style.padding = "10px";
// Resize arrow box to fit content
if (Zotero.isMac) {
let panel = document.getElementById("zotero-lookup-panel");
let box = panel.firstChild;
panel.sizeTo(box.scrollWidth, box.scrollHeight);
}
}
/**
* Focuses the field
*/
this.onShown = function (event) {
// Ignore context menu
if (event.originalTarget.id != 'zotero-lookup-panel') return;
this.getActivePanel().querySelector('textarea').focus();
}
/**
* Cancels the popup and resets fields
*/
this.onHidden = function (event) {
// Ignore context menu
if (event.originalTarget.id != 'zotero-lookup-panel') return;
document.getElementById("zotero-lookup-textbox").value = "";
Zotero_Lookup.setShowProgress(false);
// Revert to single-line when closing
this.setMultiline(false);
}
this.getActivePanel = function() {
var mlPanel = document.getElementById("zotero-lookup-multiline");
if (mlPanel.hidden) return document.getElementById("zotero-lookup-singleLine");
return mlPanel;
}
this.handleToolbarButtonMouseDown = function (event) {
var button = event.target;
if (button.disabled) {
event.preventDefault();
return;
}
this.showPanel(button)
};
/**
* Handles a key press
*/
this.onKeyPress = function (event, textBox) {
var keyCode = event.keyCode;
//use enter to start search, shift+enter to insert a new line. Flipped in multiline mode
var multiline = textBox.rows > 1;
var search = multiline ? event.shiftKey : !event.shiftKey;
if(keyCode === 13 || keyCode === 14) {
if(search) {
Zotero_Lookup.accept(textBox);
event.preventDefault();
} else if(!multiline) { // switch to multiline
Zotero_Lookup.setMultiline(true);
}
} else if(keyCode == event.DOM_VK_ESCAPE) {
document.getElementById("zotero-lookup-panel").hidePopup();
}
}
this.onInput = function (event, textBox) {
if (/[\r\n]/.test(textBox.value)) {
this.setMultiline(true);
}
};
this.setMultiline = function (on) {
var mlTxtBox = document.getElementById("zotero-lookup-textbox");
var mlButtons = document.getElementById('zotero-lookup-buttons');
mlTxtBox.rows = on ? 5 : 1;
mlButtons.hidden = !on;
// Resize arrow box to fit content -- also done in onShowing()
if(Zotero.isMac) {
var panel = document.getElementById("zotero-lookup-panel");
var box = panel.firstChild;
panel.sizeTo(box.scrollWidth, box.scrollHeight);
}
return mlTxtBox;
};
this.setShowProgress = function (on) {
// In Firefox 52.6.0, progressmeters burn CPU at idle on Linux when undetermined, even
// if they're hidden. (Being hidden is enough on macOS.)
document.getElementById("zotero-lookup-textbox").disabled = !!on;
var p = document.getElementById("zotero-lookup-multiline-progress");
if (on) {
p.removeAttribute('value');
}
else {
p.setAttribute('value', 0);
}
p.hidden = !on;
};
}