Add getSelectedObjects() and limit getSelectedItems() to items

This adds an explicit function on ZoteroPane and the item tree for
getting objects -- including collections and searches in the trash --
and limits `getSelectedItems()` to returning actual items from the
selection. We shim various item properties on the collections and
searches in the trash, but code that's getting the selection should be
explicit about what it wants. Outside of the trash, they're equivalent,
except `getSelectedObjects()` does have an `asIDs` mode.

This switches to using getSelectedObjects() in various places and fixes
collections in the trash appearing as belonging to My Publications when
using the collections-containing-an-item highlight [1].

This also adds support for highlighting the parent collections of
collections in the trash.

[1] https://forums.zotero.org/discussion/115449/zotero-7-beta-deleted-collection-appears-as-belonging-to-my-publications
This commit is contained in:
Dan Stillman 2024-06-23 04:56:19 -04:00
parent a612c1227e
commit c384fef867
4 changed files with 137 additions and 59 deletions

View file

@ -440,7 +440,7 @@ var ItemTree = class ItemTree extends LibraryTree {
var refreshed = false; var refreshed = false;
var sort = false; var sort = false;
var savedSelection = this.getSelectedItems(true); var savedSelection = this.getSelectedObjects();
var previousFirstSelectedRow = this._rowMap[ var previousFirstSelectedRow = this._rowMap[
// 'collection-item' ids are in the form <collectionID>-<itemID> // 'collection-item' ids are in the form <collectionID>-<itemID>
// 'item' events are just integers // 'item' events are just integers
@ -467,7 +467,7 @@ var ItemTree = class ItemTree extends LibraryTree {
} }
} }
// If refreshing a single item, clear caches and then deselect and reselect row // If refreshing a single item, clear caches and then deselect and reselect row
else if (savedSelection.length == 1 && savedSelection[0] == ids[0]) { else if (savedSelection.length == 1 && savedSelection[0].id == ids[0]) {
let id = ids[0]; let id = ids[0];
let row = this._rowMap[id]; let row = this._rowMap[id];
delete this._rowCache[id]; delete this._rowCache[id];
@ -802,7 +802,7 @@ var ItemTree = class ItemTree extends LibraryTree {
} }
// If single item is selected and was modified // If single item is selected and was modified
else if (action == 'modify' && ids.length == 1 && else if (action == 'modify' && ids.length == 1 &&
savedSelection.length == 1 && savedSelection[0] == ids[0]) { savedSelection.length == 1 && savedSelection[0].id == ids[0]) {
if (activeWindow) { if (activeWindow) {
await this.selectItem(ids[0]); await this.selectItem(ids[0]);
reselect = true; reselect = true;
@ -818,7 +818,7 @@ var ItemTree = class ItemTree extends LibraryTree {
|| action == 'trash' || action == 'trash'
|| action == 'delete' || action == 'delete'
|| action == 'removeDuplicatesMaster') || action == 'removeDuplicatesMaster')
&& savedSelection.some(id => this.getRowIndexByID(id) === false)) { && savedSelection.some(o => this.getRowIndexByID(o.id) === false)) {
// In duplicates view, select the next set on delete // In duplicates view, select the next set on delete
if (collectionTreeRow.isDuplicates()) { if (collectionTreeRow.isDuplicates()) {
if (this._rows[previousFirstSelectedRow]) { if (this._rows[previousFirstSelectedRow]) {
@ -1087,7 +1087,7 @@ var ItemTree = class ItemTree extends LibraryTree {
if (this.selection) { if (this.selection) {
this.selection.selectEventsSuppressed = true; this.selection.selectEventsSuppressed = true;
} }
const selection = this.getSelectedItems(true); const selection = this.getSelectedObjects();
await this.refresh(); await this.refresh();
clearItemsPaneMessage && this.clearItemsPaneMessage(); clearItemsPaneMessage && this.clearItemsPaneMessage();
await new Promise((resolve) => { await new Promise((resolve) => {
@ -1451,7 +1451,7 @@ var ItemTree = class ItemTree extends LibraryTree {
return collation.compareString(1, fieldA, fieldB); return collation.compareString(1, fieldA, fieldB);
} }
var savedSelection = this.getSelectedItems(true); var savedSelection = this.getSelectedObjects();
// Save open state and close containers before sorting // Save open state and close containers before sorting
var openItemIDs = this._saveOpenState(true); var openItemIDs = this._saveOpenState(true);
@ -1586,7 +1586,7 @@ var ItemTree = class ItemTree extends LibraryTree {
return; return;
} }
if (!skipRowMapRefresh) { if (!skipRowMapRefresh) {
var savedSelection = this.getSelectedItems(true); var savedSelection = this.getSelectedObjects();
} }
var count = 0; var count = 0;
@ -1650,7 +1650,7 @@ var ItemTree = class ItemTree extends LibraryTree {
return; return;
} }
var savedSelection = this.getSelectedItems(true); var savedSelection = this.getSelectedObjects();
for (var i=0; i<this.rowCount; i++) { for (var i=0; i<this.rowCount; i++) {
var id = this.getRow(i).ref.id; var id = this.getRow(i).ref.id;
if (searchParentIDs.has(id) && this.isContainer(i) && !this.isContainerOpen(i)) { if (searchParentIDs.has(id) && this.isContainer(i) && !this.isContainerOpen(i)) {
@ -1663,7 +1663,7 @@ var ItemTree = class ItemTree extends LibraryTree {
expandAllRows() { expandAllRows() {
this.selection.selectEventsSuppressed = true; this.selection.selectEventsSuppressed = true;
var selectedItems = this.getSelectedItems(true); var selectedItems = this.getSelectedObjects();
for (var i=0; i<this.rowCount; i++) { for (var i=0; i<this.rowCount; i++) {
if (this.isContainer(i) && !this.isContainerOpen(i)) { if (this.isContainer(i) && !this.isContainerOpen(i)) {
this.toggleOpenState(i, true); this.toggleOpenState(i, true);
@ -1678,7 +1678,7 @@ var ItemTree = class ItemTree extends LibraryTree {
collapseAllRows() { collapseAllRows() {
this.selection.selectEventsSuppressed = true; this.selection.selectEventsSuppressed = true;
const selectedItems = this.getSelectedItems(true); const selectedItems = this.getSelectedObjects();
for (var i=0; i<this.rowCount; i++) { for (var i=0; i<this.rowCount; i++) {
if (this.isContainer(i)) { if (this.isContainer(i)) {
this._closeContainer(i, true); this._closeContainer(i, true);
@ -1693,7 +1693,7 @@ var ItemTree = class ItemTree extends LibraryTree {
expandSelectedRows() { expandSelectedRows() {
this.selection.selectEventsSuppressed = true; this.selection.selectEventsSuppressed = true;
const selectedItems = this.getSelectedItems(true); const selectedItems = this.getSelectedObjects();
// Reverse sort so we don't mess up indices of subsequent // Reverse sort so we don't mess up indices of subsequent
// items when expanding // items when expanding
const indices = Array.from(this.selection.selected).sort((a, b) => b - a); const indices = Array.from(this.selection.selected).sort((a, b) => b - a);
@ -1711,7 +1711,7 @@ var ItemTree = class ItemTree extends LibraryTree {
collapseSelectedRows() { collapseSelectedRows() {
this.selection.selectEventsSuppressed = true; this.selection.selectEventsSuppressed = true;
const selectedItems = this.getSelectedItems(true); const selectedItems = this.getSelectedObjects();
// Reverse sort and so we don't mess up indices of subsequent // Reverse sort and so we don't mess up indices of subsequent
// items when collapsing // items when collapsing
const indices = Array.from(this.selection.selected).sort((a, b) => b - a); const indices = Array.from(this.selection.selected).sort((a, b) => b - a);
@ -1825,21 +1825,27 @@ var ItemTree = class ItemTree extends LibraryTree {
} }
} }
getSelectedItems(asIDs) { /**
var items = this.selection ? Array.from(this.selection.selected) : []; * Get selected objects, including collections and searches in the trash
items = items.filter(index => index < this._rows.length); */
getSelectedObjects() {
var indexes = this.selection ? Array.from(this.selection.selected) : [];
indexes = indexes.filter(index => index < this._rows.length);
try { try {
if (asIDs) return items.map(index => this.getRow(index).ref.treeViewID); return indexes.map(index => this.getRow(index).ref);
return items.map(index => this.getRow(index).ref); }
} catch (e) { catch (e) {
Zotero.debug(items); Zotero.debug(indexes);
throw e; throw e;
} }
} }
saveSelection() { /**
Zotero.debug("ItemTree::saveSelection() is deprecated -- use getSelectedItems(true)"); * Get selected items, omitting collections and searches in the trash
return this.getSelectedItems(true); */
getSelectedItems(asIDs) {
var items = this.getSelectedObjects().filter(x => x.isItem());
return asIDs ? items.map(x => x.id) : items;
} }
/** /**
@ -3127,7 +3133,7 @@ var ItemTree = class ItemTree extends LibraryTree {
if (!this.isContainerOpen(index)) return; if (!this.isContainerOpen(index)) return;
if (!skipRowMapRefresh) { if (!skipRowMapRefresh) {
var savedSelection = this.getSelectedItems(true); var savedSelection = this.getSelectedObjects();
} }
var count = 0; var count = 0;
@ -3663,12 +3669,12 @@ var ItemTree = class ItemTree extends LibraryTree {
}).bind(this); }).bind(this);
try { try {
for (let i = 0; i < selection.length; i++) { for (let i = 0; i < selection.length; i++) {
if (this._rowMap[selection[i]] != null) { if (this._rowMap[selection[i].treeViewID] != null) {
toggleSelect(selection[i]); toggleSelect(selection[i].treeViewID);
} }
// Try the parent // Try the parent
else { else {
var item = Zotero.Items.get(selection[i]); let item = selection[i];
if (!item) { if (!item) {
continue; continue;
} }
@ -3682,7 +3688,7 @@ var ItemTree = class ItemTree extends LibraryTree {
if (expandCollapsedParents) { if (expandCollapsedParents) {
await this._closeContainer(this._rowMap[parent]); await this._closeContainer(this._rowMap[parent]);
await this.toggleOpenState(this._rowMap[parent]); await this.toggleOpenState(this._rowMap[parent]);
toggleSelect(selection[i]); toggleSelect(selection[i].treeViewID);
} }
else { else {
!this.selection.isSelected(this._rowMap[parent]) && !this.selection.isSelected(this._rowMap[parent]) &&

View file

@ -47,7 +47,6 @@ var ZoteroPane = new function()
this.handleKeyDown = handleKeyDown; this.handleKeyDown = handleKeyDown;
this.captureKeyDown = captureKeyDown; this.captureKeyDown = captureKeyDown;
this.handleKeyUp = handleKeyUp; this.handleKeyUp = handleKeyUp;
this.setHighlightedRowsCallback = setHighlightedRowsCallback;
this.handleKeyPress = handleKeyPress; this.handleKeyPress = handleKeyPress;
this.getSelectedCollection = getSelectedCollection; this.getSelectedCollection = getSelectedCollection;
this.getSelectedSavedSearch = getSelectedSavedSearch; this.getSelectedSavedSearch = getSelectedSavedSearch;
@ -1102,7 +1101,7 @@ var ZoteroPane = new function()
createInstance(Components.interfaces.nsITimer); createInstance(Components.interfaces.nsITimer);
// {} implements nsITimerCallback // {} implements nsITimerCallback
this.highlightTimer.initWithCallback({ this.highlightTimer.initWithCallback({
notify: ZoteroPane_Local.setHighlightedRowsCallback notify: () => this._setHighlightedRowsCallback()
}, 225, Components.interfaces.nsITimer.TYPE_ONE_SHOT); }, 225, Components.interfaces.nsITimer.TYPE_ONE_SHOT);
} }
// If anything but Ctlr/Options was pressed, most likely a different shortcut using Ctlr/Options // If anything but Ctlr/Options was pressed, most likely a different shortcut using Ctlr/Options
@ -1227,26 +1226,40 @@ var ZoteroPane = new function()
* Highlights collections containing selected items on Ctrl (Win) or * Highlights collections containing selected items on Ctrl (Win) or
* Option/Alt (Mac/Linux) press * Option/Alt (Mac/Linux) press
*/ */
function setHighlightedRowsCallback() { this._setHighlightedRowsCallback = async function () {
var itemIDs = ZoteroPane_Local.getSelectedItems(true); var objects = this.getSelectedObjects();
// If no items or an unreasonable number, don't try
if (!itemIDs || !itemIDs.length || itemIDs.length > 100) return;
Zotero.Promise.coroutine(function* () { // If no items or an unreasonable number, don't try
var collectionIDs = yield Zotero.Collections.getCollectionsContainingItems(itemIDs, true); if (!objects.length || objects.length > 100) return;
var ids = collectionIDs.map(id => "C" + id);
var userLibraryID = Zotero.Libraries.userLibraryID; var collections = objects.filter(o => o.isCollection());
var allInPublications = Zotero.Items.get(itemIDs).every((item) => { var items = objects.filter(o => o.isItem());
return item.libraryID == userLibraryID && item.inPublications;
}) // Get parent collections of collections
if (allInPublications) { var toHighlight = [];
ids.push("P" + Zotero.Libraries.userLibraryID); for (let collection of collections) {
if (collection.parentID) {
toHighlight.push(collection.parentID);
} }
if (ids.length) { }
ZoteroPane_Local.collectionsView.setHighlightedRows(ids); // Get collections containing items
} toHighlight.push(...await Zotero.Collections.getCollectionsContainingItems(
})(); items.map(x => x.id),
} true
));
var treeViewIDs = toHighlight.map(id => 'C' + id);
var userLibraryID = Zotero.Libraries.userLibraryID;
// If no collections selected and every item is in My Publications, highlight that
var allInPublications = !collections.length && items.every((item) => {
return item.libraryID == userLibraryID && item.inPublications;
});
if (allInPublications) {
treeViewIDs.push("P" + Zotero.Libraries.userLibraryID);
}
if (treeViewIDs.length) {
await this.collectionsView.setHighlightedRows(treeViewIDs);
}
};
function handleKeyPress(event) { function handleKeyPress(event) {
@ -1916,7 +1929,7 @@ var ZoteroPane = new function()
return false; return false;
} }
var selectedItems = this.itemsView.getSelectedItems(); var selectedItems = this.itemsView.getSelectedObjects();
// Display buttons at top of item pane depending on context. This needs to run even if the // Display buttons at top of item pane depending on context. This needs to run even if the
// selection hasn't changed, because the selected items might have been modified. // selection hasn't changed, because the selected items might have been modified.
@ -2383,7 +2396,7 @@ var ZoteroPane = new function()
return false; return false;
} }
return this.getSelectedItems().some(item => item.deleted); return this.getSelectedObjects().some(o => o.deleted);
}; };
@ -2391,12 +2404,12 @@ var ZoteroPane = new function()
* @return {Promise} * @return {Promise}
*/ */
this.restoreSelectedItems = async function () { this.restoreSelectedItems = async function () {
let selectedIDs = this.getSelectedItems(true); let selectedObjects = this.getSelectedObjects();
if (!selectedIDs.length) { if (!selectedObjects.length) {
return; return;
} }
let isSelected = itemOrID => (itemOrID.treeViewID ? selectedIDs.includes(itemOrID.treeViewID) : selectedIDs.includes(itemOrID)); let isSelected = object => selectedObjects.includes(object);
await Zotero.DB.executeTransaction(async () => { await Zotero.DB.executeTransaction(async () => {
for (let row = 0; row < this.itemsView.rowCount; row++) { for (let row = 0; row < this.itemsView.rowCount; row++) {
@ -2406,7 +2419,7 @@ var ZoteroPane = new function()
} }
let parent = this.itemsView.getRow(row).ref; let parent = this.itemsView.getRow(row).ref;
let children = []; let childIDs = [];
let subcollections = []; let subcollections = [];
if (parent.isCollection()) { if (parent.isCollection()) {
// If the restored item is a collection, restore its subcollections too // If the restored item is a collection, restore its subcollections too
@ -2415,17 +2428,22 @@ var ZoteroPane = new function()
} }
} }
else { else {
if (!parent.isNote()) children.push(...parent.getNotes(true)); if (!parent.isNote()) {
if (!parent.isAttachment()) children.push(...parent.getAttachments(true)); childIDs.push(...parent.getNotes(true));
}
if (!parent.isAttachment()) {
childIDs.push(...parent.getAttachments(true));
}
} }
let childItems = Zotero.Items.get(childIDs);
if (isSelected(parent)) { if (isSelected(parent)) {
if (parent.deleted) { if (parent.deleted) {
parent.deleted = false; parent.deleted = false;
await parent.save(); await parent.save();
} }
let noneSelected = !children.some(isSelected); let noneSelected = !childItems.some(isSelected);
let allChildren = Zotero.Items.get(children).concat(Zotero.Collections.get(subcollections)); let allChildren = childItems.concat(Zotero.Collections.get(subcollections));
for (let child of allChildren) { for (let child of allChildren) {
if ((noneSelected || isSelected(child)) && child.deleted) { if ((noneSelected || isSelected(child)) && child.deleted) {
child.deleted = false; child.deleted = false;
@ -2434,7 +2452,7 @@ var ZoteroPane = new function()
} }
} }
else { else {
for (let child of Zotero.Items.get(children)) { for (let child of childItems) {
if (isSelected(child) && child.deleted) { if (isSelected(child) && child.deleted) {
child.deleted = false; child.deleted = false;
await child.save(); await child.save();
@ -2995,6 +3013,12 @@ var ZoteroPane = new function()
} }
this.getSelectedObjects = function () {
if (!this.itemsView) return [];
return this.itemsView.getSelectedObjects();
};
/* /*
* Return an array of Item objects for selected items * Return an array of Item objects for selected items
* *

View file

@ -1544,4 +1544,27 @@ describe("Zotero.ItemTree", function() {
assert.equal(OS.Path.basename(path), originalFileName); assert.equal(OS.Path.basename(path), originalFileName);
}); });
}); });
describe("#_restoreSelection()", function () {
it("should reselect collection in trash", async function () {
var userLibraryID = Zotero.Libraries.userLibraryID;
var collection = await createDataObject('collection', { deleted: true });
var item = await createDataObject('item', { deleted: true });
await cv.selectByID("T" + userLibraryID);
await waitForItemsLoad(win);
var collectionRow = zp.itemsView.getRowIndexByID(collection.treeViewID)
var itemRow = zp.itemsView.getRowIndexByID(item.id)
zp.itemsView.selection.toggleSelect(collectionRow);
zp.itemsView.selection.toggleSelect(itemRow);
var selection = zp.itemsView.getSelectedObjects();
assert.lengthOf(selection, 2);
zp.itemsView.selection.clearSelection();
assert.lengthOf(zp.itemsView.getSelectedObjects(), 0);
zp.itemsView._restoreSelection(selection);
assert.lengthOf(zp.itemsView.getSelectedObjects(), 2);
});
});
}) })

View file

@ -15,6 +15,31 @@ describe("ZoteroPane", function() {
win.close(); win.close();
}); });
describe("#_setHighlightedRowsCallback()", function () {
it("should highlight parent collection of collection in trash", async function () {
var collection1 = await createDataObject('collection');
var collection2 = await createDataObject('collection', { parentID: collection1.id, deleted: true });
var userLibraryID = Zotero.Libraries.userLibraryID;
await zp.collectionsView.selectByID('T' + userLibraryID);
await waitForItemsLoad(win);
var row = zp.itemsView.getRowIndexByID(collection2.treeViewID);
zp.itemsView.selection.select(row);
var spy = sinon.spy(zp.collectionsView, 'setHighlightedRows');
await zp._setHighlightedRowsCallback();
assert.sameMembers(spy.getCall(0).args[0], ['C1']);
var rows = win.document.querySelectorAll('.highlighted');
assert.lengthOf(rows, 1);
zp.collectionsView.setHighlightedRows();
spy.restore();
});
});
describe("#newItem", function () { describe("#newItem", function () {
it("should create an item and focus the title field", function* () { it("should create an item and focus the title field", function* () {
yield zp.newItem(Zotero.ItemTypes.getID('book'), {}, null, true); yield zp.newItem(Zotero.ItemTypes.getID('book'), {}, null, true);