2015-10-29 07:41:54 +00:00
|
|
|
"use strict";
|
|
|
|
|
2015-05-04 06:00:52 +00:00
|
|
|
describe("ZoteroPane", function() {
|
2016-03-12 10:03:47 +00:00
|
|
|
var win, doc, zp, userLibraryID;
|
2015-05-04 06:00:52 +00:00
|
|
|
|
|
|
|
// Load Zotero pane and select library
|
|
|
|
before(function* () {
|
|
|
|
win = yield loadZoteroPane();
|
|
|
|
doc = win.document;
|
|
|
|
zp = win.ZoteroPane;
|
2016-03-12 10:03:47 +00:00
|
|
|
userLibraryID = Zotero.Libraries.userLibraryID;
|
2015-05-04 06:00:52 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
after(function () {
|
|
|
|
win.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("#newItem", function () {
|
|
|
|
it("should create an item and focus the title field", function* () {
|
|
|
|
yield zp.newItem(Zotero.ItemTypes.getID('book'), {}, null, true);
|
2024-01-12 15:07:19 +00:00
|
|
|
let title = doc.getElementById('zotero-item-pane-header').querySelector("editable-text");
|
|
|
|
assert.equal(doc.activeElement.getAttribute("aria-label"), title.getAttribute("aria-label"));
|
|
|
|
title.blur();
|
2015-05-04 06:00:52 +00:00
|
|
|
yield Zotero.Promise.delay(1);
|
|
|
|
})
|
|
|
|
|
|
|
|
it("should save an entered value when New Item is used", function* () {
|
|
|
|
var value = "Test";
|
|
|
|
var item = yield zp.newItem(Zotero.ItemTypes.getID('book'), {}, null, true);
|
2024-01-12 15:07:19 +00:00
|
|
|
let header = doc.getElementById('zotero-item-pane-header');
|
|
|
|
let title = header.querySelector("editable-text");
|
|
|
|
title.value = value;
|
|
|
|
yield header.save();
|
2015-05-04 06:00:52 +00:00
|
|
|
item = yield Zotero.Items.getAsync(item.id);
|
|
|
|
assert.equal(item.getField('title'), value);
|
|
|
|
})
|
|
|
|
});
|
2015-05-08 20:01:25 +00:00
|
|
|
|
2015-05-26 01:47:06 +00:00
|
|
|
describe("#newNote()", function () {
|
|
|
|
it("should create a child note and select it", function* () {
|
|
|
|
var item = yield createDataObject('item');
|
|
|
|
var noteID = yield zp.newNote(false, item.key, "Test");
|
|
|
|
var selected = zp.itemsView.getSelectedItems(true);
|
|
|
|
assert.lengthOf(selected, 1);
|
|
|
|
assert.equal(selected, noteID);
|
|
|
|
})
|
2015-11-15 22:41:21 +00:00
|
|
|
|
|
|
|
it("should create a standalone note within a collection and select it", function* () {
|
|
|
|
var collection = yield createDataObject('collection');
|
|
|
|
var noteID = yield zp.newNote(false, false, "Test");
|
|
|
|
assert.equal(zp.collectionsView.getSelectedCollection(), collection);
|
|
|
|
var selected = zp.itemsView.getSelectedItems(true);
|
|
|
|
assert.lengthOf(selected, 1);
|
|
|
|
assert.equal(selected, noteID);
|
|
|
|
})
|
2015-05-26 01:47:06 +00:00
|
|
|
})
|
|
|
|
|
2016-02-03 06:13:30 +00:00
|
|
|
describe("#newCollection()", function () {
|
|
|
|
it("should create a collection", function* () {
|
2024-01-16 19:45:16 +00:00
|
|
|
var promise = waitForDialog(
|
|
|
|
null,
|
|
|
|
'accept',
|
|
|
|
'chrome://zotero/content/newCollectionDialog.xhtml'
|
|
|
|
);
|
2016-02-03 06:13:30 +00:00
|
|
|
var id = yield zp.newCollection();
|
|
|
|
yield promise;
|
|
|
|
var collection = Zotero.Collections.get(id);
|
|
|
|
assert.isTrue(collection.name.startsWith(Zotero.getString('pane.collections.untitled')));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-09-24 01:02:36 +00:00
|
|
|
describe("#newSearch()", function () {
|
|
|
|
it("should create a saved search", function* () {
|
|
|
|
var promise = waitForDialog(
|
|
|
|
// TODO: Test changing a condition
|
|
|
|
function (dialog) {},
|
|
|
|
'accept',
|
2022-07-04 05:48:52 +00:00
|
|
|
'chrome://zotero/content/searchDialog.xhtml'
|
2016-09-24 01:02:36 +00:00
|
|
|
);
|
|
|
|
var id = yield zp.newSearch();
|
|
|
|
yield promise;
|
|
|
|
var search = Zotero.Searches.get(id);
|
|
|
|
assert.ok(search);
|
|
|
|
assert.isTrue(search.name.startsWith(Zotero.getString('pane.collections.untitled')));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle clicking Cancel in the search window", function* () {
|
|
|
|
var promise = waitForDialog(
|
|
|
|
function (dialog) {},
|
|
|
|
'cancel',
|
2022-07-04 05:48:52 +00:00
|
|
|
'chrome://zotero/content/searchDialog.xhtml'
|
2016-09-24 01:02:36 +00:00
|
|
|
);
|
|
|
|
var id = yield zp.newSearch();
|
|
|
|
yield promise;
|
|
|
|
assert.isFalse(id);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-05-08 20:01:25 +00:00
|
|
|
describe("#itemSelected()", function () {
|
2015-05-19 07:55:54 +00:00
|
|
|
it.skip("should update the item count", function* () {
|
2015-05-08 20:01:25 +00:00
|
|
|
var collection = new Zotero.Collection;
|
|
|
|
collection.name = "Count Test";
|
2015-05-10 08:20:47 +00:00
|
|
|
var id = yield collection.saveTx();
|
2015-05-08 20:01:25 +00:00
|
|
|
yield waitForItemsLoad(win);
|
|
|
|
|
|
|
|
// Unselected, with no items in view
|
|
|
|
assert.equal(
|
2016-03-24 12:55:32 +00:00
|
|
|
doc.getElementById('zotero-item-pane-message-box').textContent,
|
2015-05-08 20:01:25 +00:00
|
|
|
Zotero.getString('pane.item.unselected.zero', 0)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Unselected, with one item in view
|
|
|
|
var item = new Zotero.Item('newspaperArticle');
|
|
|
|
item.setCollections([id]);
|
2015-05-10 08:20:47 +00:00
|
|
|
var itemID1 = yield item.saveTx({
|
2015-05-08 20:01:25 +00:00
|
|
|
skipSelect: true
|
|
|
|
});
|
|
|
|
assert.equal(
|
2016-03-24 12:55:32 +00:00
|
|
|
doc.getElementById('zotero-item-pane-message-box').textContent,
|
2015-05-08 20:01:25 +00:00
|
|
|
Zotero.getString('pane.item.unselected.singular', 1)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Unselected, with multiple items in view
|
|
|
|
var item = new Zotero.Item('audioRecording');
|
|
|
|
item.setCollections([id]);
|
2015-05-10 08:20:47 +00:00
|
|
|
var itemID2 = yield item.saveTx({
|
2015-05-08 20:01:25 +00:00
|
|
|
skipSelect: true
|
|
|
|
});
|
|
|
|
assert.equal(
|
2016-03-24 12:55:32 +00:00
|
|
|
doc.getElementById('zotero-item-pane-message-box').textContent,
|
2015-05-08 20:01:25 +00:00
|
|
|
Zotero.getString('pane.item.unselected.plural', 2)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Multiple items selected
|
|
|
|
var promise = zp.itemsView._getItemSelectedPromise();
|
|
|
|
zp.itemsView.rememberSelection([itemID1, itemID2]);
|
|
|
|
yield promise;
|
|
|
|
assert.equal(
|
2016-03-24 12:55:32 +00:00
|
|
|
doc.getElementById('zotero-item-pane-message-box').textContent,
|
2015-05-08 20:01:25 +00:00
|
|
|
Zotero.getString('pane.item.selected.multiple', 2)
|
|
|
|
);
|
|
|
|
})
|
|
|
|
})
|
2015-10-29 07:41:54 +00:00
|
|
|
|
|
|
|
describe("#viewAttachment", function () {
|
|
|
|
var apiKey = Zotero.Utilities.randomString(24);
|
2023-07-02 02:58:38 +00:00
|
|
|
var baseURL;
|
2017-08-15 23:42:21 +00:00
|
|
|
var httpd;
|
2015-10-29 07:41:54 +00:00
|
|
|
|
2017-08-15 23:42:21 +00:00
|
|
|
async function downloadOnDemand() {
|
|
|
|
var item = new Zotero.Item("attachment");
|
|
|
|
item.attachmentLinkMode = 'imported_file';
|
|
|
|
item.attachmentPath = 'storage:test.txt';
|
|
|
|
// TODO: Test binary data
|
|
|
|
var text = Zotero.Utilities.randomString();
|
|
|
|
item.attachmentSyncState = "to_download";
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var mtime = "1441252524000";
|
|
|
|
var md5 = Zotero.Utilities.Internal.md5(text)
|
|
|
|
|
|
|
|
var s3Path = `pretend-s3/${item.key}`;
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
`/users/1/items/${item.key}/file`,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 302, "Found");
|
|
|
|
response.setHeader("Zotero-File-Modification-Time", mtime, false);
|
|
|
|
response.setHeader("Zotero-File-MD5", md5, false);
|
|
|
|
response.setHeader("Zotero-File-Compressed", "No", false);
|
|
|
|
response.setHeader("Location", baseURL + s3Path, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
"/" + s3Path,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// Disable loadURI() so viewAttachment() doesn't trigger translator loading
|
2018-08-19 08:31:40 +00:00
|
|
|
var stub = sinon.stub(Zotero, "launchFile");
|
2017-08-15 23:42:21 +00:00
|
|
|
|
|
|
|
await zp.viewAttachment(item.id);
|
|
|
|
|
|
|
|
assert.ok(stub.calledOnce);
|
2018-08-19 08:31:40 +00:00
|
|
|
assert.ok(stub.calledWith(item.getFilePath()));
|
2017-08-15 23:42:21 +00:00
|
|
|
stub.restore();
|
|
|
|
|
|
|
|
assert.equal(await item.attachmentHash, md5);
|
|
|
|
assert.equal(await item.attachmentModificationTime, mtime);
|
|
|
|
var path = await item.getFilePathAsync();
|
|
|
|
assert.equal(await Zotero.File.getContentsAsync(path), text);
|
|
|
|
};
|
|
|
|
|
2015-10-29 07:41:54 +00:00
|
|
|
before(function () {
|
|
|
|
Zotero.HTTP.mock = sinon.FakeXMLHttpRequest;
|
|
|
|
})
|
2023-08-16 05:10:56 +00:00
|
|
|
beforeEach(async function () {
|
|
|
|
var port;
|
|
|
|
({ httpd, port } = await startHTTPServer());
|
|
|
|
baseURL = `http://localhost:${port}/`;
|
2017-08-15 23:42:21 +00:00
|
|
|
Zotero.Prefs.set("api.url", baseURL);
|
2015-10-29 07:41:54 +00:00
|
|
|
|
2023-07-02 02:58:38 +00:00
|
|
|
Zotero.Sync.Runner.apiKey = apiKey;
|
2023-08-16 05:10:56 +00:00
|
|
|
await Zotero.Users.setCurrentUserID(1);
|
|
|
|
await Zotero.Users.setCurrentUsername("testuser");
|
2015-10-29 07:41:54 +00:00
|
|
|
})
|
|
|
|
afterEach(function* () {
|
|
|
|
var defer = new Zotero.Promise.defer();
|
2017-08-15 23:42:21 +00:00
|
|
|
httpd.stop(() => defer.resolve());
|
2015-10-29 07:41:54 +00:00
|
|
|
yield defer.promise;
|
|
|
|
})
|
2017-08-15 23:42:21 +00:00
|
|
|
after(function () {
|
|
|
|
Zotero.HTTP.mock = null;
|
|
|
|
});
|
2015-10-29 07:41:54 +00:00
|
|
|
|
2017-08-15 23:42:21 +00:00
|
|
|
it("should download an attachment on-demand in as-needed mode", function* () {
|
|
|
|
Zotero.Sync.Storage.Local.downloadAsNeeded(Zotero.Libraries.userLibraryID, true);
|
|
|
|
yield downloadOnDemand();
|
|
|
|
});
|
|
|
|
|
2020-03-07 07:25:44 +00:00
|
|
|
// As noted in viewAttachment(), this is only necessary for files modified before 5.0.85
|
|
|
|
it("should re-download a remotely modified attachment in as-needed mode", async function () {
|
|
|
|
Zotero.Sync.Storage.Local.downloadAsNeeded(Zotero.Libraries.userLibraryID, true);
|
|
|
|
|
|
|
|
var item = await importFileAttachment('test.txt');
|
|
|
|
item.attachmentSyncState = "to_download";
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var text = Zotero.Utilities.randomString();
|
|
|
|
var mtime = "1441252524000";
|
|
|
|
var md5 = Zotero.Utilities.Internal.md5(text)
|
|
|
|
|
|
|
|
var s3Path = `pretend-s3/${item.key}`;
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
`/users/1/items/${item.key}/file`,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 302, "Found");
|
|
|
|
response.setHeader("Zotero-File-Modification-Time", mtime, false);
|
|
|
|
response.setHeader("Zotero-File-MD5", md5, false);
|
|
|
|
response.setHeader("Zotero-File-Compressed", "No", false);
|
|
|
|
response.setHeader("Location", baseURL + s3Path, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
"/" + s3Path,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// Disable loadURI() so viewAttachment() doesn't trigger translator loading
|
|
|
|
var downloadSpy = sinon.spy(Zotero.Sync.Runner, "downloadFile");
|
|
|
|
var launchFileStub = sinon.stub(Zotero, "launchFile");
|
|
|
|
|
|
|
|
await zp.viewAttachment(item.id);
|
|
|
|
|
|
|
|
assert.ok(downloadSpy.calledOnce);
|
|
|
|
assert.ok(launchFileStub.calledOnce);
|
|
|
|
assert.ok(launchFileStub.calledWith(item.getFilePath()));
|
|
|
|
downloadSpy.restore();
|
|
|
|
launchFileStub.restore();
|
|
|
|
|
|
|
|
assert.equal(await item.attachmentHash, md5);
|
|
|
|
assert.equal(await item.attachmentModificationTime, mtime);
|
|
|
|
var path = await item.getFilePathAsync();
|
|
|
|
assert.equal(await Zotero.File.getContentsAsync(path), text);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle a 404 when re-downloading a remotely modified attachment in as-needed mode", async function () {
|
|
|
|
Zotero.Sync.Storage.Local.downloadAsNeeded(Zotero.Libraries.userLibraryID, true);
|
|
|
|
|
|
|
|
var item = await importFileAttachment('test.txt');
|
|
|
|
item.attachmentSyncState = "to_download";
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var mtime = await item.attachmentModificationTime;
|
|
|
|
var md5 = await item.attachmentHash;
|
|
|
|
var text = await Zotero.File.getContentsAsync(item.getFilePath());
|
|
|
|
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
`/users/1/items/${item.key}/file`,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 404, "Not Found");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// Disable loadURI() so viewAttachment() doesn't trigger translator loading
|
|
|
|
var downloadSpy = sinon.spy(Zotero.Sync.Runner, "downloadFile");
|
|
|
|
var launchFileStub = sinon.stub(Zotero, "launchFile");
|
|
|
|
|
|
|
|
await zp.viewAttachment(item.id);
|
|
|
|
|
|
|
|
assert.ok(downloadSpy.calledOnce);
|
|
|
|
assert.ok(launchFileStub.calledOnce);
|
|
|
|
assert.ok(launchFileStub.calledWith(item.getFilePath()));
|
|
|
|
downloadSpy.restore();
|
|
|
|
launchFileStub.restore();
|
|
|
|
|
|
|
|
// File shouldn't have changed
|
|
|
|
assert.equal(await item.attachmentModificationTime, mtime);
|
|
|
|
assert.equal(await item.attachmentHash, md5);
|
|
|
|
var path = await item.getFilePathAsync();
|
|
|
|
assert.equal(await Zotero.File.getContentsAsync(path), text);
|
|
|
|
});
|
|
|
|
|
2017-08-15 23:42:21 +00:00
|
|
|
it("should download an attachment on-demand in at-sync-time mode", function* () {
|
|
|
|
Zotero.Sync.Storage.Local.downloadOnSync(Zotero.Libraries.userLibraryID, true);
|
|
|
|
yield downloadOnDemand();
|
|
|
|
});
|
2023-09-23 07:09:23 +00:00
|
|
|
|
|
|
|
it("should update a PDF with a blank MIME type", async function () {
|
|
|
|
let attachment = await importFileAttachment('test.pdf');
|
|
|
|
// Can't use contentType argument to importFileAttachment() because blank string is ignored
|
|
|
|
attachment.attachmentContentType = '';
|
|
|
|
await attachment.saveTx();
|
|
|
|
await zp.viewAttachment(attachment.id);
|
|
|
|
assert.equal(attachment.attachmentContentType, 'application/pdf');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should update an EPUB with an 'application/epub' MIME type", async function () {
|
|
|
|
let attachment = await importFileAttachment('stub.epub', { contentType: 'application/epub' });
|
|
|
|
assert.equal(attachment.attachmentContentType, 'application/epub');
|
|
|
|
await zp.viewAttachment(attachment.id);
|
|
|
|
assert.equal(attachment.attachmentContentType, 'application/epub+zip');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should update an EPUB with an 'application/octet-stream' MIME type", async function () {
|
|
|
|
let attachment = await importFileAttachment('stub.epub', { contentType: 'application/octet-stream' });
|
|
|
|
assert.equal(attachment.attachmentContentType, 'application/octet-stream');
|
|
|
|
await zp.viewAttachment(attachment.id);
|
|
|
|
assert.equal(attachment.attachmentContentType, 'application/epub+zip');
|
|
|
|
});
|
2015-10-29 07:41:54 +00:00
|
|
|
})
|
2016-03-12 10:03:47 +00:00
|
|
|
|
2017-04-12 04:56:37 +00:00
|
|
|
|
2022-09-01 23:46:59 +00:00
|
|
|
describe("#addNoteFromAnnotationsFromSelected()", function () {
|
|
|
|
it("should create a single note within a selected regular item for all child attachments", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var attachment1 = await importPDFAttachment(item);
|
|
|
|
var attachment2 = await importPDFAttachment(item);
|
|
|
|
var annotation1 = await createAnnotation('highlight', attachment1);
|
|
|
|
var annotation2 = await createAnnotation('highlight', attachment1);
|
|
|
|
var annotation3 = await createAnnotation('highlight', attachment2);
|
|
|
|
var annotation4 = await createAnnotation('highlight', attachment2);
|
|
|
|
await zp.selectItems([item.id]);
|
|
|
|
await zp.addNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
assert.equal(note.itemType, 'note');
|
|
|
|
assert.equal(note.parentID, item.id);
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h3')].map(x => x.textContent),
|
|
|
|
[attachment1.attachmentFilename, attachment2.attachmentFilename]
|
|
|
|
);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3 + p')], 2);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 4);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should create a single note within the parent for all selected sibling attachments", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var attachment1 = await importPDFAttachment(item);
|
|
|
|
var attachment2 = await importPDFAttachment(item);
|
|
|
|
var annotation1 = await createAnnotation('highlight', attachment1);
|
|
|
|
var annotation2 = await createAnnotation('highlight', attachment1);
|
|
|
|
var annotation3 = await createAnnotation('highlight', attachment2);
|
|
|
|
var annotation4 = await createAnnotation('highlight', attachment2);
|
|
|
|
await zp.selectItems([attachment1.id, attachment2.id]);
|
|
|
|
await zp.addNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
assert.equal(note.parentID, item.id);
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h3')].map(x => x.textContent),
|
|
|
|
[attachment1.attachmentFilename, attachment2.attachmentFilename]
|
|
|
|
);
|
|
|
|
// No item titles
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h2 + p')], 0);
|
|
|
|
// Just attachment titles
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3 + p')], 2);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 4);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should ignore top-level item if child attachment is also selected", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var attachment1 = await importPDFAttachment(item);
|
|
|
|
var attachment2 = await importPDFAttachment(item);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await zp.selectItems([item.id, attachment1.id]);
|
|
|
|
await zp.addNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
// No titles
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h2 + p')], 0);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3 + p')], 0);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("shouldn't do anything if parent item and child note is selected", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var attachment = await importPDFAttachment(item);
|
|
|
|
var note = await createDataObject('item', { itemType: 'note', parentID: item.id });
|
|
|
|
await createAnnotation('highlight', attachment);
|
|
|
|
await zp.selectItems([item.id, note.id]);
|
|
|
|
await zp.addNoteFromAnnotationsFromSelected();
|
|
|
|
var selectedItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(selectedItems, 2);
|
|
|
|
assert.sameMembers(selectedItems, [item, note]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("#createStandaloneNoteFromAnnotationsFromSelected()", function () {
|
|
|
|
it("should create a single standalone note for all child attachments of selected regular items", async function () {
|
|
|
|
var collection = await createDataObject('collection');
|
|
|
|
var item1 = await createDataObject('item', { setTitle: true, collections: [collection.id] });
|
|
|
|
var item2 = await createDataObject('item', { setTitle: true, collections: [collection.id] });
|
|
|
|
var attachment1 = await importPDFAttachment(item1);
|
|
|
|
var attachment2 = await importPDFAttachment(item1);
|
|
|
|
var attachment3 = await importPDFAttachment(item2);
|
|
|
|
var attachment4 = await importPDFAttachment(item2);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await zp.selectItems([item1.id, item2.id]);
|
|
|
|
await zp.createStandaloneNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
assert.equal(note.itemType, 'note');
|
|
|
|
assert.isFalse(note.parentID);
|
|
|
|
assert.isTrue(collection.hasItem(note));
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h2')].map(x => x.textContent),
|
|
|
|
[item1.getDisplayTitle(), item2.getDisplayTitle()]
|
|
|
|
);
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h3')].map(x => x.textContent),
|
|
|
|
[
|
|
|
|
attachment1.attachmentFilename,
|
|
|
|
attachment2.attachmentFilename,
|
|
|
|
attachment3.attachmentFilename,
|
|
|
|
attachment4.attachmentFilename
|
|
|
|
]
|
|
|
|
);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3 + p')], 4);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 8);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should create a single standalone note for all selected attachments", async function () {
|
|
|
|
var collection = await createDataObject('collection');
|
|
|
|
var item1 = await createDataObject('item', { setTitle: true, collections: [collection.id] });
|
|
|
|
var item2 = await createDataObject('item', { setTitle: true, collections: [collection.id] });
|
|
|
|
var attachment1 = await importPDFAttachment(item1);
|
|
|
|
var attachment2 = await importPDFAttachment(item1);
|
|
|
|
var attachment3 = await importPDFAttachment(item2);
|
|
|
|
var attachment4 = await importPDFAttachment(item2);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await zp.selectItems([attachment1.id, attachment3.id]);
|
|
|
|
await zp.createStandaloneNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
assert.isFalse(note.parentID);
|
|
|
|
assert.isTrue(collection.hasItem(note));
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h2')].map(x => x.textContent),
|
|
|
|
[item1.getDisplayTitle(), item2.getDisplayTitle()]
|
|
|
|
);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h2 + p')], 2);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3')], 0);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 4);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should ignore top-level item if child attachment is also selected", async function () {
|
|
|
|
var item1 = await createDataObject('item', { setTitle: true });
|
|
|
|
var item2 = await createDataObject('item', { setTitle: true });
|
|
|
|
var attachment1 = await importPDFAttachment(item1);
|
|
|
|
var attachment2 = await importPDFAttachment(item1);
|
|
|
|
var attachment3 = await importPDFAttachment(item2);
|
|
|
|
var attachment4 = await importPDFAttachment(item2);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment1);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment2);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment3);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await createAnnotation('highlight', attachment4);
|
|
|
|
await zp.selectItems([item1.id, attachment1.id, attachment3.id]);
|
|
|
|
await zp.createStandaloneNoteFromAnnotationsFromSelected();
|
|
|
|
var newItems = zp.getSelectedItems();
|
|
|
|
assert.lengthOf(newItems, 1);
|
|
|
|
var note = newItems[0];
|
|
|
|
var dp = new DOMParser();
|
|
|
|
var doc = dp.parseFromString(note.getNote(), 'text/html');
|
|
|
|
assert.sameMembers(
|
|
|
|
[...doc.querySelectorAll('h2')].map(x => x.textContent),
|
|
|
|
[item1.getDisplayTitle(), item2.getDisplayTitle()]
|
|
|
|
);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h2 + p')], 2);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('h3')], 0);
|
|
|
|
assert.lengthOf([...doc.querySelectorAll('span.highlight')], 4);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2017-11-01 05:02:23 +00:00
|
|
|
describe("#renameSelectedAttachmentsFromParents()", function () {
|
|
|
|
it("should rename a linked file", async function () {
|
|
|
|
var oldFilename = 'old.png';
|
|
|
|
var newFilename = 'Test.png';
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var tmpDir = await getTempDirectory();
|
|
|
|
var oldFile = OS.Path.join(tmpDir, oldFilename);
|
|
|
|
await OS.File.copy(file.path, oldFile);
|
|
|
|
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: oldFile,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, newFilename);
|
|
|
|
var path = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(path), newFilename)
|
|
|
|
await OS.File.exists(path);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should use unique name for linked file if target name is taken", async function () {
|
|
|
|
var oldFilename = 'old.png';
|
|
|
|
var newFilename = 'Test.png';
|
|
|
|
var uniqueFilename = 'Test 2.png';
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var tmpDir = await getTempDirectory();
|
|
|
|
var oldFile = OS.Path.join(tmpDir, oldFilename);
|
|
|
|
await OS.File.copy(file.path, oldFile);
|
|
|
|
// Create file with target filename
|
|
|
|
await Zotero.File.putContentsAsync(OS.Path.join(tmpDir, newFilename), '');
|
|
|
|
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: oldFile,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, uniqueFilename);
|
|
|
|
var path = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(path), uniqueFilename)
|
|
|
|
await OS.File.exists(path);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should use unique name for linked file without extension if target name is taken", async function () {
|
|
|
|
var oldFilename = 'old';
|
|
|
|
var newFilename = 'Test';
|
|
|
|
var uniqueFilename = 'Test 2';
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var tmpDir = await getTempDirectory();
|
|
|
|
var oldFile = OS.Path.join(tmpDir, oldFilename);
|
|
|
|
await OS.File.copy(file.path, oldFile);
|
|
|
|
// Create file with target filename
|
|
|
|
await Zotero.File.putContentsAsync(OS.Path.join(tmpDir, newFilename), '');
|
|
|
|
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: oldFile,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, uniqueFilename);
|
|
|
|
var path = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(path), uniqueFilename)
|
|
|
|
await OS.File.exists(path);
|
|
|
|
});
|
2023-07-25 05:44:42 +00:00
|
|
|
|
|
|
|
it("shouldn't change attachment title if different from filename", async function () {
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Title');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await importFileAttachment('test.png', { parentItemID: item.id });
|
|
|
|
attachment.setField('title', 'Image');
|
|
|
|
await attachment.saveTx();
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
assert.isTrue(await zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, 'Title.png');
|
|
|
|
assert.equal(attachment.getField('title'), 'Image')
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should change attachment title if the same as filename", async function () {
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Title');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await importFileAttachment('test.png', { parentItemID: item.id });
|
|
|
|
attachment.setField('title', 'test.png');
|
|
|
|
await attachment.saveTx();
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
assert.isTrue(await zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, 'Title.png');
|
|
|
|
assert.equal(attachment.getField('title'), 'Title.png')
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should change attachment title if the same as filename without extension", async function () {
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.setField('title', 'Title');
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var attachment = await importFileAttachment('test.png', { parentItemID: item.id });
|
|
|
|
attachment.setField('title', 'test');
|
|
|
|
await attachment.saveTx();
|
|
|
|
await zp.selectItem(attachment.id);
|
|
|
|
|
|
|
|
assert.isTrue(await zp.renameSelectedAttachmentsFromParents());
|
|
|
|
assert.equal(attachment.attachmentFilename, 'Title.png');
|
|
|
|
assert.equal(attachment.getField('title'), 'Title.png')
|
|
|
|
});
|
2017-11-01 05:02:23 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
|
2017-10-02 02:04:11 +00:00
|
|
|
describe("#duplicateSelectedItem()", function () {
|
|
|
|
it("should add reverse relations", async function () {
|
2017-10-02 02:42:51 +00:00
|
|
|
await selectLibrary(win);
|
2017-10-02 02:04:11 +00:00
|
|
|
var item1 = await createDataObject('item');
|
|
|
|
var item2 = await createDataObject('item');
|
|
|
|
item1.addRelatedItem(item2);
|
|
|
|
await item1.saveTx();
|
|
|
|
item2.addRelatedItem(item1);
|
|
|
|
await item2.saveTx();
|
|
|
|
var item3 = await zp.duplicateSelectedItem();
|
|
|
|
assert.sameMembers(item3.relatedItems, [item1.key]);
|
|
|
|
assert.sameMembers(item2.relatedItems, [item1.key]);
|
|
|
|
assert.sameMembers(item1.relatedItems, [item2.key, item3.key]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2022-08-13 09:45:23 +00:00
|
|
|
describe("#duplicateAndConvertSelectedItem()", function () {
|
|
|
|
describe("book to book section", function () {
|
|
|
|
it("should not add relations to other book sections for the same book", async function () {
|
|
|
|
await selectLibrary(win);
|
|
|
|
var bookItem = await createDataObject('item', { itemType: 'book', title: "Book Title" });
|
|
|
|
|
|
|
|
// Relate book to another book section with a different title
|
|
|
|
var otherBookSection = createUnsavedDataObject('item', { itemType: 'bookSection', setTitle: true })
|
|
|
|
otherBookSection.setField('bookTitle', "Another Book Title");
|
|
|
|
await otherBookSection.saveTx();
|
|
|
|
bookItem.addRelatedItem(otherBookSection);
|
|
|
|
await bookItem.saveTx();
|
|
|
|
otherBookSection.addRelatedItem(bookItem);
|
|
|
|
await otherBookSection.saveTx();
|
|
|
|
|
|
|
|
await zp.selectItem(bookItem.id);
|
|
|
|
var bookSectionItem1 = await zp.duplicateAndConvertSelectedItem();
|
|
|
|
await zp.selectItem(bookItem.id);
|
|
|
|
var bookSectionItem2 = await zp.duplicateAndConvertSelectedItem();
|
|
|
|
|
|
|
|
// Book sections should only be related to parent
|
|
|
|
assert.sameMembers(bookSectionItem1.relatedItems, [bookItem.key, otherBookSection.key]);
|
|
|
|
assert.sameMembers(bookSectionItem2.relatedItems, [bookItem.key, otherBookSection.key]);
|
|
|
|
});
|
|
|
|
});
|
2022-08-29 20:15:31 +00:00
|
|
|
|
|
|
|
it("should not copy abstracts", async function() {
|
|
|
|
await selectLibrary(win);
|
|
|
|
var bookItem = await createDataObject('item', { itemType: 'book', title: "Book Title" });
|
|
|
|
bookItem.setField('abstractNote', 'An abstract');
|
|
|
|
bookItem.saveTx();
|
|
|
|
|
|
|
|
var bookSectionItem = await zp.duplicateAndConvertSelectedItem();
|
|
|
|
assert.isEmpty(bookSectionItem.getField('abstractNote'));
|
|
|
|
});
|
2022-08-13 09:45:23 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
|
2017-04-12 04:56:37 +00:00
|
|
|
describe("#deleteSelectedItems()", function () {
|
2021-12-21 08:19:13 +00:00
|
|
|
const DELETE_KEY_CODE = 46;
|
|
|
|
|
2017-04-12 04:56:37 +00:00
|
|
|
it("should remove an item from My Publications", function* () {
|
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.inPublications = true;
|
|
|
|
yield item.saveTx();
|
|
|
|
|
|
|
|
yield zp.collectionsView.selectByID("P" + userLibraryID);
|
|
|
|
yield waitForItemsLoad(win);
|
|
|
|
var iv = zp.itemsView;
|
|
|
|
|
|
|
|
var selected = iv.selectItem(item.id);
|
|
|
|
assert.ok(selected);
|
|
|
|
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
var tree = doc.getElementById(iv.id);
|
2017-04-12 04:56:37 +00:00
|
|
|
tree.focus();
|
|
|
|
|
|
|
|
yield Zotero.Promise.delay(1);
|
|
|
|
|
|
|
|
var promise = waitForDialog();
|
|
|
|
var modifyPromise = waitForItemEvent('modify');
|
|
|
|
|
2022-06-21 07:24:48 +00:00
|
|
|
var event = new KeyboardEvent(
|
2021-12-21 08:19:13 +00:00
|
|
|
"keypress",
|
2022-06-21 07:24:48 +00:00
|
|
|
{
|
|
|
|
key: 'Delete',
|
|
|
|
code: 'Delete',
|
|
|
|
keyCode: DELETE_KEY_CODE,
|
|
|
|
bubbles: true,
|
|
|
|
cancelable: true
|
|
|
|
}
|
2021-12-21 08:19:13 +00:00
|
|
|
);
|
2017-04-12 04:56:37 +00:00
|
|
|
tree.dispatchEvent(event);
|
|
|
|
yield promise;
|
|
|
|
yield modifyPromise;
|
|
|
|
|
|
|
|
assert.isFalse(item.inPublications);
|
|
|
|
assert.isFalse(item.deleted);
|
|
|
|
});
|
|
|
|
|
2021-12-21 08:19:13 +00:00
|
|
|
it("should move My Publications item to trash with prompt for modified Delete", function* () {
|
2017-04-12 04:56:37 +00:00
|
|
|
var item = createUnsavedDataObject('item');
|
|
|
|
item.inPublications = true;
|
|
|
|
yield item.saveTx();
|
|
|
|
|
|
|
|
yield zp.collectionsView.selectByID("P" + userLibraryID);
|
|
|
|
yield waitForItemsLoad(win);
|
|
|
|
var iv = zp.itemsView;
|
|
|
|
|
|
|
|
var selected = iv.selectItem(item.id);
|
|
|
|
assert.ok(selected);
|
|
|
|
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
var tree = doc.getElementById(iv.id);
|
2017-04-12 04:56:37 +00:00
|
|
|
tree.focus();
|
|
|
|
|
|
|
|
yield Zotero.Promise.delay(1);
|
|
|
|
|
|
|
|
var promise = waitForDialog();
|
|
|
|
var modifyPromise = waitForItemEvent('modify');
|
|
|
|
|
2022-06-21 07:24:48 +00:00
|
|
|
var event = new KeyboardEvent(
|
2017-04-12 04:56:37 +00:00
|
|
|
"keypress",
|
2022-06-21 07:24:48 +00:00
|
|
|
{
|
|
|
|
key: 'Delete',
|
|
|
|
code: 'Delete',
|
|
|
|
keyCode: DELETE_KEY_CODE,
|
|
|
|
bubbles: true,
|
|
|
|
cancelable: true,
|
|
|
|
shiftKey: !Zotero.isMac,
|
|
|
|
metaKey: Zotero.isMac,
|
|
|
|
}
|
2017-04-12 04:56:37 +00:00
|
|
|
);
|
|
|
|
tree.dispatchEvent(event);
|
|
|
|
yield promise;
|
|
|
|
yield modifyPromise;
|
|
|
|
|
|
|
|
assert.isTrue(item.inPublications);
|
|
|
|
assert.isTrue(item.deleted);
|
|
|
|
});
|
2021-12-21 08:19:13 +00:00
|
|
|
|
|
|
|
it("should move saved search item to trash with prompt for unmodified Delete", async function () {
|
|
|
|
var search = await createDataObject('search');
|
|
|
|
var title = [...Object.values(search.conditions)]
|
|
|
|
.filter(x => x.condition == 'title' && x.operator == 'contains')[0].value;
|
|
|
|
var item = await createDataObject('item', { title });
|
|
|
|
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
var iv = zp.itemsView;
|
|
|
|
|
|
|
|
var selected = iv.selectItem(item.id);
|
|
|
|
assert.ok(selected);
|
|
|
|
|
|
|
|
var tree = doc.getElementById(iv.id);
|
|
|
|
tree.focus();
|
|
|
|
|
|
|
|
await Zotero.Promise.delay(1);
|
|
|
|
|
|
|
|
var promise = waitForDialog();
|
|
|
|
var modifyPromise = waitForItemEvent('modify');
|
|
|
|
|
|
|
|
var event = new KeyboardEvent(
|
|
|
|
"keypress",
|
|
|
|
{
|
|
|
|
key: 'Delete',
|
|
|
|
code: 'Delete',
|
|
|
|
keyCode: DELETE_KEY_CODE,
|
|
|
|
bubbles: true,
|
|
|
|
cancelable: true
|
|
|
|
}
|
|
|
|
);
|
|
|
|
tree.dispatchEvent(event);
|
|
|
|
await promise;
|
|
|
|
await modifyPromise;
|
|
|
|
|
|
|
|
assert.isTrue(item.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should move saved search trash without prompt for modified Delete", async function () {
|
|
|
|
var search = await createDataObject('search');
|
|
|
|
var title = [...Object.values(search.conditions)]
|
|
|
|
.filter(x => x.condition == 'title' && x.operator == 'contains')[0].value;
|
|
|
|
var item = await createDataObject('item', { title });
|
|
|
|
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
var iv = zp.itemsView;
|
|
|
|
|
|
|
|
var selected = iv.selectItem(item.id);
|
|
|
|
assert.ok(selected);
|
|
|
|
|
|
|
|
var tree = doc.getElementById(iv.id);
|
|
|
|
tree.focus();
|
|
|
|
|
|
|
|
await Zotero.Promise.delay(1);
|
|
|
|
|
|
|
|
var modifyPromise = waitForItemEvent('modify');
|
|
|
|
|
|
|
|
var event = new KeyboardEvent(
|
|
|
|
"keypress",
|
|
|
|
{
|
|
|
|
key: 'Delete',
|
|
|
|
code: 'Delete',
|
2022-06-21 07:24:48 +00:00
|
|
|
keyCode: DELETE_KEY_CODE,
|
2021-12-21 08:19:13 +00:00
|
|
|
metaKey: Zotero.isMac,
|
|
|
|
shiftKey: !Zotero.isMac,
|
|
|
|
bubbles: true,
|
|
|
|
cancelable: true
|
|
|
|
}
|
|
|
|
);
|
|
|
|
tree.dispatchEvent(event);
|
|
|
|
await modifyPromise;
|
|
|
|
|
|
|
|
assert.isTrue(item.deleted);
|
|
|
|
});
|
2022-07-09 05:37:14 +00:00
|
|
|
|
|
|
|
it("should prompt to remove an item from subcollections when recursiveCollections enabled", async function () {
|
|
|
|
Zotero.Prefs.set('recursiveCollections', true);
|
|
|
|
|
|
|
|
let collection1 = await createDataObject('collection');
|
|
|
|
let collection2 = await createDataObject('collection', { parentID: collection1.id });
|
|
|
|
let item = await createDataObject('item', { collections: [collection2.id] });
|
|
|
|
assert.ok(await zp.collectionsView.selectCollection(collection1.id));
|
|
|
|
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
|
|
|
|
let iv = zp.itemsView;
|
|
|
|
assert.ok(await iv.selectItem(item.id));
|
|
|
|
|
|
|
|
await Zotero.Promise.delay(1);
|
|
|
|
|
|
|
|
let promise = waitForDialog();
|
|
|
|
let modifyPromise = waitForItemEvent('modify');
|
|
|
|
|
|
|
|
await zp.deleteSelectedItems(false);
|
|
|
|
|
|
|
|
let dialog = await promise;
|
|
|
|
await modifyPromise;
|
|
|
|
|
|
|
|
assert.include(dialog.document.documentElement.textContent, Zotero.getString('pane.items.removeRecursive'));
|
|
|
|
assert.isFalse(item.inCollection(collection2.id));
|
|
|
|
|
|
|
|
Zotero.Prefs.clear('recursiveCollections');
|
|
|
|
});
|
2017-04-12 04:56:37 +00:00
|
|
|
});
|
|
|
|
|
2016-09-27 06:10:14 +00:00
|
|
|
describe("#deleteSelectedCollection()", function () {
|
|
|
|
it("should delete collection but not descendant items by default", function* () {
|
|
|
|
var collection = yield createDataObject('collection');
|
|
|
|
var item = yield createDataObject('item', { collections: [collection.id] });
|
|
|
|
var promise = waitForDialog();
|
|
|
|
yield zp.deleteSelectedCollection();
|
|
|
|
assert.isFalse(Zotero.Collections.exists(collection.id));
|
|
|
|
assert.isTrue(Zotero.Items.exists(item.id));
|
|
|
|
assert.isFalse(item.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should delete collection and descendant items when deleteItems=true", function* () {
|
|
|
|
var collection = yield createDataObject('collection');
|
|
|
|
var item = yield createDataObject('item', { collections: [collection.id] });
|
|
|
|
var promise = waitForDialog();
|
|
|
|
yield zp.deleteSelectedCollection(true);
|
|
|
|
assert.isFalse(Zotero.Collections.exists(collection.id));
|
|
|
|
assert.isTrue(Zotero.Items.exists(item.id));
|
|
|
|
assert.isTrue(item.deleted);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-03-12 10:03:47 +00:00
|
|
|
|
|
|
|
describe("#setVirtual()", function () {
|
|
|
|
var cv;
|
|
|
|
|
|
|
|
before(function* () {
|
|
|
|
cv = zp.collectionsView;
|
|
|
|
});
|
|
|
|
beforeEach(function () {
|
|
|
|
Zotero.Prefs.clear('duplicateLibraries');
|
|
|
|
Zotero.Prefs.clear('unfiledLibraries');
|
|
|
|
return selectLibrary(win);
|
|
|
|
})
|
|
|
|
|
2016-07-01 09:48:19 +00:00
|
|
|
it("should show a hidden virtual collection in My Library", function* () {
|
2016-03-14 00:31:15 +00:00
|
|
|
// Create unfiled, duplicate items
|
|
|
|
var title = Zotero.Utilities.randomString();
|
|
|
|
var item1 = yield createDataObject('item', { title });
|
|
|
|
var item2 = yield createDataObject('item', { title });
|
|
|
|
|
2016-07-01 09:48:19 +00:00
|
|
|
// Start hidden (tested in collectionTreeViewTest)
|
|
|
|
Zotero.Prefs.set('duplicateLibraries', `{"${userLibraryID}": false}`);
|
|
|
|
Zotero.Prefs.set('unfiledLibraries', `{"${userLibraryID}": false}`);
|
2016-03-12 10:03:47 +00:00
|
|
|
yield cv.refresh();
|
|
|
|
|
|
|
|
// Show Duplicate Items
|
|
|
|
var id = "D" + userLibraryID;
|
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2019-06-10 06:37:54 +00:00
|
|
|
yield zp.setVirtual(userLibraryID, 'duplicates', true, true);
|
2016-07-01 09:48:19 +00:00
|
|
|
// Duplicate Items should be selected
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(zp.getCollectionTreeRow().id, id);
|
2016-07-01 09:48:19 +00:00
|
|
|
// Should be missing from pref
|
|
|
|
assert.isUndefined(JSON.parse(Zotero.Prefs.get('duplicateLibraries'))[userLibraryID])
|
2016-03-14 00:31:15 +00:00
|
|
|
|
|
|
|
// Clicking should select both items
|
|
|
|
var row = cv.getRowIndexByID(id);
|
|
|
|
assert.ok(row);
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(cv.selection.pivot, row);
|
2016-03-14 00:31:15 +00:00
|
|
|
yield waitForItemsLoad(win);
|
|
|
|
var iv = zp.itemsView;
|
|
|
|
row = iv.getRowIndexByID(item1.id);
|
|
|
|
assert.isNumber(row);
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
var promise = iv.waitForSelect();
|
|
|
|
clickOnItemsRow(win, iv, row);
|
2016-03-14 00:31:15 +00:00
|
|
|
assert.equal(iv.selection.count, 2);
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
yield promise;
|
2016-03-12 10:03:47 +00:00
|
|
|
|
|
|
|
// Show Unfiled Items
|
|
|
|
id = "U" + userLibraryID;
|
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2019-06-10 06:37:54 +00:00
|
|
|
yield zp.setVirtual(userLibraryID, 'unfiled', true, true);
|
2016-07-01 09:48:19 +00:00
|
|
|
// Unfiled Items should be selected
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(zp.getCollectionTreeRow().id, id);
|
2016-07-01 09:48:19 +00:00
|
|
|
// Should be missing from pref
|
|
|
|
assert.isUndefined(JSON.parse(Zotero.Prefs.get('unfiledLibraries'))[userLibraryID])
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should expand library if collapsed when showing virtual collection", function* () {
|
|
|
|
// Start hidden (tested in collectionTreeViewTest)
|
|
|
|
Zotero.Prefs.set('duplicateLibraries', `{"${userLibraryID}": false}`);
|
|
|
|
yield cv.refresh();
|
|
|
|
|
|
|
|
var libraryRow = cv.getRowIndexByID(Zotero.Libraries.userLibrary.treeViewID);
|
|
|
|
if (cv.isContainerOpen(libraryRow)) {
|
|
|
|
yield cv.toggleOpenState(libraryRow);
|
|
|
|
cv._saveOpenStates();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Show Duplicate Items
|
|
|
|
var id = "D" + userLibraryID;
|
2019-06-10 06:37:54 +00:00
|
|
|
yield zp.setVirtual(userLibraryID, 'duplicates', true, true);
|
2016-07-01 09:48:19 +00:00
|
|
|
|
|
|
|
// Library should have been expanded and Duplicate Items selected
|
2016-03-12 10:03:47 +00:00
|
|
|
assert.ok(cv.getRowIndexByID(id));
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(zp.getCollectionTreeRow().id, id);
|
2016-03-12 10:03:47 +00:00
|
|
|
});
|
|
|
|
|
2016-07-01 09:48:19 +00:00
|
|
|
it("should hide a virtual collection in My Library", function* () {
|
2016-03-12 10:03:47 +00:00
|
|
|
yield cv.refresh();
|
|
|
|
|
|
|
|
// Hide Duplicate Items
|
|
|
|
var id = "D" + userLibraryID;
|
|
|
|
assert.ok(yield cv.selectByID(id));
|
|
|
|
yield zp.setVirtual(userLibraryID, 'duplicates', false);
|
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2016-07-01 09:48:19 +00:00
|
|
|
assert.isFalse(JSON.parse(Zotero.Prefs.get('duplicateLibraries'))[userLibraryID])
|
2016-03-12 10:03:47 +00:00
|
|
|
|
|
|
|
// Hide Unfiled Items
|
|
|
|
id = "U" + userLibraryID;
|
|
|
|
assert.ok(yield cv.selectByID(id));
|
|
|
|
yield zp.setVirtual(userLibraryID, 'unfiled', false);
|
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2016-07-01 09:48:19 +00:00
|
|
|
assert.isFalse(JSON.parse(Zotero.Prefs.get('unfiledLibraries'))[userLibraryID])
|
2016-03-12 10:03:47 +00:00
|
|
|
});
|
|
|
|
|
2016-07-01 09:48:19 +00:00
|
|
|
it("should hide a virtual collection in a group", function* () {
|
2016-03-12 10:03:47 +00:00
|
|
|
yield cv.refresh();
|
|
|
|
|
2016-07-01 09:48:19 +00:00
|
|
|
var group = yield createGroup();
|
|
|
|
var groupRow = cv.getRowIndexByID(group.treeViewID);
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
var rowCount = cv._rows.length;
|
2016-07-01 09:48:19 +00:00
|
|
|
|
|
|
|
// Make sure group is open
|
|
|
|
if (!cv.isContainerOpen(groupRow)) {
|
|
|
|
yield cv.toggleOpenState(groupRow);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure Duplicate Items is showing
|
|
|
|
var id = "D" + group.libraryID;
|
|
|
|
assert.ok(cv.getRowIndexByID(id));
|
|
|
|
|
2016-03-12 10:03:47 +00:00
|
|
|
// Hide Duplicate Items
|
|
|
|
assert.ok(yield cv.selectByID(id));
|
2016-07-01 09:48:19 +00:00
|
|
|
yield zp.setVirtual(group.libraryID, 'duplicates', false);
|
|
|
|
// Row should have been removed
|
2016-03-12 10:03:47 +00:00
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2016-07-01 09:48:19 +00:00
|
|
|
// Pref should have been updated
|
|
|
|
Zotero.debug(Zotero.Prefs.get('duplicateLibraries'));
|
|
|
|
assert.isFalse(JSON.parse(Zotero.Prefs.get('duplicateLibraries'))[group.libraryID]);
|
|
|
|
// Group row shouldn't have changed
|
|
|
|
assert.equal(cv.getRowIndexByID(group.treeViewID), groupRow);
|
|
|
|
// Group should remain open
|
|
|
|
assert.isTrue(cv.isContainerOpen(groupRow));
|
|
|
|
// Row count should be 1 less
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(cv._rows.length, --rowCount);
|
2016-03-12 10:03:47 +00:00
|
|
|
|
|
|
|
// Hide Unfiled Items
|
2016-07-01 09:48:19 +00:00
|
|
|
id = "U" + group.libraryID;
|
2016-03-12 10:03:47 +00:00
|
|
|
assert.ok(yield cv.selectByID(id));
|
2016-07-01 09:48:19 +00:00
|
|
|
// Hide Unfiled Items
|
|
|
|
yield zp.setVirtual(group.libraryID, 'unfiled', false);
|
|
|
|
// Row should have been removed
|
2016-03-12 10:03:47 +00:00
|
|
|
assert.isFalse(cv.getRowIndexByID(id));
|
2019-01-21 01:39:27 +00:00
|
|
|
// Pref should have been updated
|
2016-07-01 09:48:19 +00:00
|
|
|
assert.isFalse(JSON.parse(Zotero.Prefs.get('unfiledLibraries'))[group.libraryID]);
|
|
|
|
// Group row shouldn't have changed
|
|
|
|
assert.equal(cv.getRowIndexByID(group.treeViewID), groupRow);
|
|
|
|
// Group should remain open
|
|
|
|
assert.isTrue(cv.isContainerOpen(groupRow));
|
|
|
|
// Row count should be 1 less
|
XUL -> JS tree megacommit
- Just a single huge commit. This has been developed over too long a
time, required many tiny changes across too many files and has seen too
many iterations to be separated into separate commits.
The original branch with all the messy commits will be kept around for
posterity
https://github.com/zotero/zotero/compare/bb220ad0f2d6bf0eca6df6d225d3d358cb50a27b...adomasven:feature/react-item-tree
- Replaces XUL <tree> element across the whole zotero client codebase
with a custom supermegafast virtualized-table inspired by
react-virtualized yet mimicking old XUL treeview API. The
virtualized-table sits on top on a raw-to-the-metal,
interpreted-at-runtime JS based windowing solution inspired by
react-window. React-based solutions could not be used because they were
slow and Zotero UI needs to be responsive and be able to
display thousands of rows in a treeview without any slowdowns.
- Attempts were made at making this screen-reader friendly, but yet to
be tested with something like JAWS
- RTL-friendly
- Styling and behaviour across all platforms was copied as closely as
possible to the original XUL tree
- Instead of row-based scroll snapping this has smooth-scrolling. If
you're using arrow keys to browse through the tree then it effectively
snap-scrolls. Current CSS snap scroll attributes do not seem to work in
the way we would require even on up-to-date browsers, yet alone the ESR
version of FX that Zotero is on. JS solutions are either terrible for
performance or produce inexcusable jitter.
- When dragging-and-dropping items the initial drag freezes the UI for
a fairly jarring amount of time. Does not seem to be fixable due to
the synchronous code that needs to be run in the dragstart handler.
Used to be possible to run that code async with the XUL tree.
- Item tree column picker no longer has a dedicated button. Just
right-click the columns. The column preferences (width, order, etc) are
no longer handled by XUL, which required a custom serialization and
storage solution that throws warnings in the developer console due to
the amount of data being stored. Might cause temporary freezing on HDDs
upon column resize/reorder/visibility toggling.
- Context menu handling code basically unchanged, but any UI changes
that plugins may have wanted to do (including adding new columns) will
have to be redone by them. No serious thought has gone into how plugin
developers would achieve that yet.
- Opens up the possibility for awesome alternative ways to render the
tree items, including things like multiple-row view for the item tree,
which has been requested for a long while especially by users switching
from other referencing software
2020-06-03 07:29:46 +00:00
|
|
|
assert.equal(cv._rows.length, --rowCount);
|
2016-03-12 10:03:47 +00:00
|
|
|
});
|
|
|
|
});
|
2016-03-26 06:59:54 +00:00
|
|
|
|
|
|
|
describe("#editSelectedCollection()", function () {
|
|
|
|
it("should edit a saved search", function* () {
|
|
|
|
var search = yield createDataObject('search');
|
2022-07-04 05:48:52 +00:00
|
|
|
var promise = waitForWindow('chrome://zotero/content/searchDialog.xhtml', function (win) {
|
2016-03-26 06:59:54 +00:00
|
|
|
let searchBox = win.document.getElementById('search-box');
|
|
|
|
var c = searchBox.search.getCondition(
|
|
|
|
searchBox.search.addCondition("title", "contains", "foo")
|
|
|
|
);
|
|
|
|
searchBox.addCondition(c);
|
2023-04-29 08:24:01 +00:00
|
|
|
win.document.querySelector('dialog').acceptDialog();
|
2016-03-26 06:59:54 +00:00
|
|
|
});
|
|
|
|
yield zp.editSelectedCollection();
|
2016-10-21 02:55:42 +00:00
|
|
|
yield promise;
|
|
|
|
var conditions = search.getConditions();
|
|
|
|
assert.lengthOf(Object.keys(conditions), 3);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should edit a saved search in a group", function* () {
|
|
|
|
var group = yield getGroup();
|
|
|
|
var search = yield createDataObject('search', { libraryID: group.libraryID });
|
2022-07-04 05:48:52 +00:00
|
|
|
var promise = waitForWindow('chrome://zotero/content/searchDialog.xhtml', function (win) {
|
2016-10-21 02:55:42 +00:00
|
|
|
let searchBox = win.document.getElementById('search-box');
|
|
|
|
var c = searchBox.search.getCondition(
|
|
|
|
searchBox.search.addCondition("title", "contains", "foo")
|
|
|
|
);
|
|
|
|
searchBox.addCondition(c);
|
2023-04-29 08:24:01 +00:00
|
|
|
win.document.querySelector('dialog').acceptDialog();
|
2016-10-21 02:55:42 +00:00
|
|
|
});
|
|
|
|
yield zp.editSelectedCollection();
|
2016-03-26 06:59:54 +00:00
|
|
|
yield promise;
|
|
|
|
var conditions = search.getConditions();
|
2016-07-07 12:39:19 +00:00
|
|
|
assert.lengthOf(Object.keys(conditions), 3);
|
2016-03-26 06:59:54 +00:00
|
|
|
});
|
|
|
|
});
|
2021-12-27 04:15:47 +00:00
|
|
|
|
|
|
|
describe("#buildItemContextMenu()", function () {
|
2021-12-27 04:50:15 +00:00
|
|
|
it("shouldn't show export or bib options for multiple standalone file attachments without notes", async function () {
|
2021-12-27 04:15:47 +00:00
|
|
|
var item1 = await importFileAttachment('test.png');
|
|
|
|
var item2 = await importFileAttachment('test.png');
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id, item2.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
2021-12-27 04:50:15 +00:00
|
|
|
|
|
|
|
var menu = win.document.getElementById('zotero-itemmenu');
|
|
|
|
assert.isTrue(menu.querySelector('.zotero-menuitem-export').hidden);
|
|
|
|
assert.isTrue(menu.querySelector('.zotero-menuitem-create-bibliography').hidden);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should show “Export Note…” for standalone file attachment with note", async function () {
|
|
|
|
var item1 = await importFileAttachment('test.png');
|
|
|
|
item1.setNote('<p>Foo</p>');
|
|
|
|
await item1.saveTx();
|
|
|
|
var item2 = await importFileAttachment('test.png');
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id, item2.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
|
|
|
|
var menu = win.document.getElementById('zotero-itemmenu');
|
|
|
|
var exportMenuItem = menu.querySelector('.zotero-menuitem-export');
|
|
|
|
assert.isFalse(exportMenuItem.hidden);
|
|
|
|
assert.equal(
|
|
|
|
exportMenuItem.getAttribute('label'),
|
|
|
|
Zotero.getString('pane.items.menu.exportNote.multiple')
|
|
|
|
);
|
2021-12-27 04:15:47 +00:00
|
|
|
});
|
2022-02-20 16:45:53 +00:00
|
|
|
|
|
|
|
it("should enable “Delete Item…” when selected item or an ancestor is in trash", async function () {
|
|
|
|
var item1 = await createDataObject('item', { deleted: true });
|
|
|
|
var attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
|
|
|
|
await zp.selectItems([attachment1.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
var menu = win.document.getElementById('zotero-itemmenu');
|
|
|
|
var deleteMenuItem = menu.querySelector('.zotero-menuitem-delete-from-lib');
|
|
|
|
assert.isFalse(deleteMenuItem.disabled);
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id, attachment1.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
assert.isFalse(deleteMenuItem.disabled);
|
|
|
|
|
|
|
|
item1.deleted = false;
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await item1.saveTx();
|
|
|
|
await attachment1.saveTx();
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
assert.isTrue(deleteMenuItem.disabled);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should enable “Restore to Library” when at least one selected item is in trash", async function () {
|
|
|
|
var item1 = await createDataObject('item', { deleted: true });
|
|
|
|
var attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
var menu = win.document.getElementById('zotero-itemmenu');
|
|
|
|
var restoreMenuItem = menu.querySelector('.zotero-menuitem-restore-to-library');
|
|
|
|
assert.isFalse(restoreMenuItem.disabled);
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id, attachment1.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
assert.isFalse(restoreMenuItem.disabled);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should disable “Restore to Library” when no selected items are in trash", async function () {
|
|
|
|
var item1 = await createDataObject('item');
|
|
|
|
var attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
|
|
|
|
await zp.selectItems([item1.id]);
|
|
|
|
await zp.buildItemContextMenu();
|
|
|
|
var menu = win.document.getElementById('zotero-itemmenu');
|
|
|
|
var restoreMenuItem = menu.querySelector('.zotero-menuitem-restore-to-library');
|
|
|
|
assert.isTrue(restoreMenuItem.disabled);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("#restoreSelectedItems()", function () {
|
|
|
|
it("should restore trashed parent and single trashed child when both are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: true });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([item1.id, attachment1.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isFalse(attachment1.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should restore child when parent and trashed child are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: false });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([item1.id, attachment1.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isFalse(attachment1.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should restore parent and selected children when parent and some trashed children are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: false });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment2 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
attachment2.deleted = true;
|
|
|
|
await attachment2.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([item1.id, attachment1.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isFalse(attachment1.deleted);
|
|
|
|
assert.isTrue(attachment2.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should restore parent and all children when trashed parent and no children are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: true });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment2 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment3 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
attachment2.deleted = true;
|
|
|
|
await attachment2.saveTx();
|
|
|
|
attachment3.deleted = true;
|
|
|
|
await attachment3.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([item1.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isFalse(attachment1.deleted);
|
|
|
|
assert.isFalse(attachment2.deleted);
|
|
|
|
assert.isFalse(attachment3.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should restore parent and selected children when trashed parent and some trashed children are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: true });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment2 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment3 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
attachment2.deleted = true;
|
|
|
|
await attachment2.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([item1.id, attachment2.id, attachment3.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isTrue(attachment1.deleted);
|
|
|
|
assert.isFalse(attachment2.deleted);
|
|
|
|
assert.isFalse(attachment3.deleted);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should restore selected children when trashed children and untrashed children are selected", async function () {
|
|
|
|
let item1 = await createDataObject('item', { deleted: false });
|
|
|
|
let attachment1 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment2 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
let attachment3 = await importFileAttachment('test.png', { parentItemID: item1.id });
|
|
|
|
attachment1.deleted = true;
|
|
|
|
await attachment1.saveTx();
|
|
|
|
attachment2.deleted = true;
|
|
|
|
await attachment2.saveTx();
|
|
|
|
|
|
|
|
var userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
await zp.collectionsView.selectByID('T' + userLibraryID);
|
|
|
|
await zp.selectItems([attachment1.id, attachment2.id, attachment3.id]);
|
|
|
|
await zp.restoreSelectedItems();
|
|
|
|
|
|
|
|
assert.isFalse(item1.deleted);
|
|
|
|
assert.isFalse(attachment1.deleted);
|
|
|
|
assert.isFalse(attachment2.deleted);
|
|
|
|
assert.isFalse(attachment3.deleted);
|
|
|
|
});
|
2021-12-27 04:15:47 +00:00
|
|
|
});
|
2022-03-01 19:34:46 +00:00
|
|
|
|
|
|
|
describe("#checkForLinkedFilesToRelink()", function () {
|
|
|
|
let labdDir;
|
|
|
|
|
|
|
|
this.beforeEach(async () => {
|
|
|
|
labdDir = await getTempDirectory();
|
|
|
|
Zotero.Prefs.set('baseAttachmentPath', labdDir);
|
|
|
|
Zotero.Prefs.set('saveRelativeAttachmentPath', true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should detect and relink a single attachment", async function () {
|
|
|
|
let item = await createDataObject('item');
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let outsideStorageDir = await getTempDirectory();
|
|
|
|
let outsideFile = OS.Path.join(outsideStorageDir, 'test.pdf');
|
|
|
|
|
|
|
|
let labdFile = OS.Path.join(labdDir, 'test.pdf');
|
|
|
|
|
|
|
|
await OS.File.copy(file.path, outsideFile);
|
|
|
|
|
|
|
|
let attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: outsideFile,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment.fileExists());
|
|
|
|
await OS.File.move(outsideFile, labdFile);
|
|
|
|
await assert.eventually.isFalse(attachment.fileExists());
|
|
|
|
|
|
|
|
let stub = sinon.stub(zp, 'showLinkedFileFoundAutomaticallyDialog')
|
|
|
|
.returns('one');
|
|
|
|
await zp.checkForLinkedFilesToRelink(attachment);
|
|
|
|
assert.ok(stub.calledOnce);
|
|
|
|
assert.ok(stub.calledWith(attachment, sinon.match.string, 0));
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment.fileExists());
|
|
|
|
assert.equal(attachment.getFilePath(), labdFile);
|
|
|
|
assert.equal(attachment.attachmentPath, 'attachments:test.pdf');
|
|
|
|
|
|
|
|
stub.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should detect and relink multiple attachments when user chooses", async function () {
|
|
|
|
for (let choice of ['one', 'all']) {
|
|
|
|
let file1 = getTestDataDirectory();
|
|
|
|
file1.append('test.pdf');
|
|
|
|
let file2 = getTestDataDirectory();
|
|
|
|
file2.append('empty.pdf');
|
|
|
|
let outsideStorageDir = await getTempDirectory();
|
|
|
|
let outsideFile1 = OS.Path.join(outsideStorageDir, 'test.pdf');
|
|
|
|
let outsideFile2 = OS.Path.join(outsideStorageDir, 'empty.pdf');
|
|
|
|
|
|
|
|
let labdFile1 = OS.Path.join(labdDir, 'test.pdf');
|
|
|
|
let labdFile2 = OS.Path.join(labdDir, 'empty.pdf');
|
|
|
|
|
|
|
|
await OS.File.copy(file1.path, outsideFile1);
|
|
|
|
await OS.File.copy(file2.path, outsideFile2);
|
|
|
|
|
|
|
|
let attachment1 = await Zotero.Attachments.linkFromFile({ file: outsideFile1 });
|
|
|
|
let attachment2 = await Zotero.Attachments.linkFromFile({ file: outsideFile2 });
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment1.fileExists());
|
|
|
|
await assert.eventually.isTrue(attachment2.fileExists());
|
|
|
|
await OS.File.move(outsideFile1, labdFile1);
|
|
|
|
await OS.File.move(outsideFile2, labdFile2);
|
|
|
|
await assert.eventually.isFalse(attachment1.fileExists());
|
|
|
|
await assert.eventually.isFalse(attachment2.fileExists());
|
|
|
|
|
|
|
|
let stub = sinon.stub(zp, 'showLinkedFileFoundAutomaticallyDialog')
|
|
|
|
.returns(choice);
|
|
|
|
await zp.checkForLinkedFilesToRelink(attachment1);
|
|
|
|
assert.ok(stub.calledOnce);
|
|
|
|
assert.ok(stub.calledWith(attachment1, sinon.match.string, 1));
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment1.fileExists());
|
|
|
|
await assert.eventually.equal(attachment2.fileExists(), choice === 'all');
|
|
|
|
assert.equal(attachment1.getFilePath(), labdFile1);
|
|
|
|
assert.equal(attachment1.attachmentPath, 'attachments:test.pdf');
|
|
|
|
if (choice === 'all') {
|
|
|
|
assert.equal(attachment2.getFilePath(), labdFile2);
|
|
|
|
assert.equal(attachment2.attachmentPath, 'attachments:empty.pdf');
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert.equal(attachment2.getFilePath(), outsideFile2);
|
|
|
|
}
|
|
|
|
|
|
|
|
stub.restore();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should use subdirectories of original path", async function () {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let outsideStorageDir = OS.Path.join(await getTempDirectory(), 'subdir');
|
|
|
|
await OS.File.makeDir(outsideStorageDir);
|
|
|
|
let outsideFile = OS.Path.join(outsideStorageDir, 'test.pdf');
|
|
|
|
|
|
|
|
let labdSubdir = OS.Path.join(labdDir, 'subdir');
|
|
|
|
await OS.File.makeDir(labdSubdir);
|
|
|
|
let labdFile = OS.Path.join(labdSubdir, 'test.pdf');
|
|
|
|
|
|
|
|
await OS.File.copy(file.path, outsideFile);
|
|
|
|
|
|
|
|
let attachment = await Zotero.Attachments.linkFromFile({ file: outsideFile });
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment.fileExists());
|
|
|
|
await OS.File.move(outsideFile, labdFile);
|
|
|
|
await assert.eventually.isFalse(attachment.fileExists());
|
|
|
|
|
|
|
|
let dialogStub = sinon.stub(zp, 'showLinkedFileFoundAutomaticallyDialog')
|
|
|
|
.returns('one');
|
|
|
|
let existsSpy = sinon.spy(OS.File, 'exists');
|
|
|
|
await zp.checkForLinkedFilesToRelink(attachment);
|
|
|
|
assert.ok(dialogStub.calledOnce);
|
|
|
|
assert.ok(dialogStub.calledWith(attachment, sinon.match.string, 0));
|
|
|
|
assert.ok(existsSpy.calledWith(OS.Path.join(labdSubdir, 'test.pdf')));
|
|
|
|
assert.notOk(existsSpy.calledWith(OS.Path.join(labdDir, 'test.pdf'))); // Should never get there
|
|
|
|
|
|
|
|
await assert.eventually.isTrue(attachment.fileExists());
|
|
|
|
assert.equal(attachment.getFilePath(), labdFile);
|
|
|
|
assert.equal(attachment.attachmentPath, 'attachments:subdir/test.pdf');
|
|
|
|
|
|
|
|
dialogStub.restore();
|
|
|
|
existsSpy.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle Windows paths", async function () {
|
|
|
|
let filenames = [['test.pdf'], ['empty.pdf'], ['search', 'baz.pdf']];
|
|
|
|
let labdFiles = [];
|
|
|
|
let attachments = [];
|
|
|
|
|
|
|
|
for (let parts of filenames) {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
parts.forEach(part => file.append(part));
|
|
|
|
|
|
|
|
await OS.File.makeDir(OS.Path.join(labdDir, ...parts.slice(0, -1)));
|
|
|
|
let labdFile = OS.Path.join(labdDir, ...parts);
|
|
|
|
await OS.File.copy(file.path, labdFile);
|
|
|
|
labdFiles.push(labdFile);
|
|
|
|
|
|
|
|
let attachment = await Zotero.Attachments.linkFromFile({ file });
|
|
|
|
attachment.attachmentPath = `C:\\test\\${parts.join('\\')}`;
|
|
|
|
await attachment.saveTx();
|
|
|
|
attachments.push(attachment);
|
|
|
|
|
|
|
|
await assert.eventually.isFalse(attachment.fileExists());
|
|
|
|
}
|
|
|
|
|
|
|
|
let stub = sinon.stub(zp, 'showLinkedFileFoundAutomaticallyDialog')
|
|
|
|
.returns('all');
|
|
|
|
await zp.checkForLinkedFilesToRelink(attachments[0]);
|
|
|
|
assert.ok(stub.calledOnce);
|
|
|
|
assert.ok(stub.calledWith(attachments[0], sinon.match.string, filenames.length - 1));
|
|
|
|
|
|
|
|
for (let i = 0; i < filenames.length; i++) {
|
|
|
|
let attachment = attachments[i];
|
|
|
|
await assert.eventually.isTrue(attachment.fileExists());
|
|
|
|
assert.equal(attachment.getFilePath(), labdFiles[i]);
|
|
|
|
assert.equal(attachment.attachmentPath, 'attachments:' + OS.Path.join(...filenames[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
stub.restore();
|
|
|
|
});
|
|
|
|
});
|
2023-10-31 08:04:13 +00:00
|
|
|
|
|
|
|
describe("#focus()", function () {
|
|
|
|
before(async function () {
|
|
|
|
var collection = new Zotero.Collection;
|
|
|
|
collection.name = "Focus Test";
|
|
|
|
await collection.saveTx();
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
});
|
|
|
|
|
|
|
|
var tab = new KeyboardEvent('keydown', {
|
|
|
|
key: 'Tab',
|
|
|
|
shiftKey: false,
|
|
|
|
bubbles: true
|
|
|
|
});
|
|
|
|
|
|
|
|
var shiftTab = new KeyboardEvent('keydown', {
|
|
|
|
key: 'Tab',
|
|
|
|
shiftKey: true,
|
|
|
|
bubbles: true
|
|
|
|
});
|
|
|
|
|
|
|
|
var rightArrow = new KeyboardEvent('keydown', {
|
|
|
|
key: 'ArrowRight',
|
|
|
|
bubbles: true
|
|
|
|
});
|
|
|
|
var leftArrow = new KeyboardEvent('keydown', {
|
|
|
|
key: 'ArrowLeft',
|
|
|
|
bubbles: true
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it("should shift-tab through the toolbar to item-tree", async function () {
|
|
|
|
let searchBox = doc.getElementById('zotero-tb-search-textbox');
|
|
|
|
searchBox.focus();
|
|
|
|
|
|
|
|
let sequence = [
|
|
|
|
"zotero-tb-search-dropmarker",
|
|
|
|
"zotero-tb-add",
|
|
|
|
"zotero-collections-search",
|
|
|
|
"zotero-tb-collection-add",
|
|
|
|
"zotero-tb-sync",
|
2023-12-22 06:06:42 +00:00
|
|
|
"zotero-tb-tabs-menu"
|
2023-10-31 08:04:13 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
for (let id of sequence) {
|
|
|
|
doc.activeElement.dispatchEvent(shiftTab);
|
2023-11-01 18:50:50 +00:00
|
|
|
// Wait for collection search to be revealed
|
|
|
|
if (id === "zotero-collections-search") {
|
2024-01-04 10:54:22 +00:00
|
|
|
await Zotero.Promise.delay(250);
|
2023-11-01 18:50:50 +00:00
|
|
|
}
|
2023-10-31 08:04:13 +00:00
|
|
|
assert.equal(doc.activeElement.id, id);
|
2024-01-04 10:54:22 +00:00
|
|
|
// Wait for collection search to be hidden for subsequent tests
|
|
|
|
if (id === "zotero-tb-collection-add") {
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
}
|
2023-10-31 08:04:13 +00:00
|
|
|
}
|
|
|
|
doc.activeElement.dispatchEvent(shiftTab);
|
|
|
|
assert.equal(doc.activeElement.className, "tab selected");
|
|
|
|
|
|
|
|
doc.activeElement.dispatchEvent(shiftTab);
|
|
|
|
assert.equal(doc.activeElement.id, "item-tree-main-default");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should tab through the toolbar to collection-tree", async function () {
|
|
|
|
win.Zotero_Tabs.moveFocus("current");
|
|
|
|
let sequence = [
|
2023-12-22 06:06:42 +00:00
|
|
|
"zotero-tb-tabs-menu",
|
2023-10-31 08:04:13 +00:00
|
|
|
"zotero-tb-sync",
|
|
|
|
"zotero-tb-collection-add",
|
|
|
|
"zotero-collections-search",
|
|
|
|
"zotero-tb-add",
|
|
|
|
"zotero-tb-search-dropmarker",
|
|
|
|
'zotero-tb-search-textbox',
|
|
|
|
'collection-tree',
|
|
|
|
];
|
|
|
|
for (let id of sequence) {
|
|
|
|
doc.activeElement.dispatchEvent(tab);
|
2023-11-01 18:50:50 +00:00
|
|
|
// Wait for collection search to be revealed
|
|
|
|
if (id === "zotero-collections-search") {
|
2024-01-04 10:54:22 +00:00
|
|
|
await Zotero.Promise.delay(250);
|
2023-11-01 18:50:50 +00:00
|
|
|
}
|
2023-10-31 08:04:13 +00:00
|
|
|
assert.equal(doc.activeElement.id, id);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should navigate toolbarbuttons with arrows", async function () {
|
|
|
|
let addItem = doc.getElementById('zotero-tb-add');
|
|
|
|
addItem.focus();
|
|
|
|
|
|
|
|
doc.activeElement.dispatchEvent(rightArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-lookup");
|
|
|
|
doc.activeElement.dispatchEvent(rightArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-attachment-add");
|
|
|
|
doc.activeElement.dispatchEvent(rightArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-note-add");
|
|
|
|
|
|
|
|
doc.activeElement.dispatchEvent(leftArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-attachment-add");
|
|
|
|
doc.activeElement.dispatchEvent(leftArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-lookup");
|
|
|
|
doc.activeElement.dispatchEvent(leftArrow);
|
|
|
|
assert.equal(doc.activeElement.id, "zotero-tb-add");
|
|
|
|
});
|
|
|
|
});
|
2015-05-04 06:00:52 +00:00
|
|
|
})
|